节俭 - 从简单的JSON转换

时间:2013-11-28 08:54:46

标签: java json thrift thrift-protocol

我创建了以下Thrift对象:

struct Student{
        1: string id;
        2: string firstName;
        3: string lastName
}

现在我想从JSON中读取这个对象。根据这个post,这是可能的

所以我写了下面的代码:

String json = "{\"id\":\"aaa\",\"firstName\":\"Danny\",\"lastName\":\"Lesnik\"}";
    StudentThriftObject s = new StudentThriftObject();
    byte[] jsonAsByte = json.getBytes("UTF-8");
    TMemoryBuffer memBuffer = new TMemoryBuffer(jsonAsByte.length);
    memBuffer.write(jsonAsByte);

    TProtocol proto = new TJSONProtocol(memBuffer);
    s.read(proto);

我得到的是以下例外:

Exception in thread "main" org.apache.thrift.protocol.TProtocolException: Unexpected character:i
    at org.apache.thrift.protocol.TJSONProtocol.readJSONSyntaxChar(TJSONProtocol.java:322)
    at org.apache.thrift.protocol.TJSONProtocol.readJSONInteger(TJSONProtocol.java:698)
    at org.apache.thrift.protocol.TJSONProtocol.readFieldBegin(TJSONProtocol.java:837)
    at com.vanilla.thrift.example.entities.StudentThriftObject$StudentThriftObjectStandardScheme.read(StudentThriftObject.java:486)
    at com.vanilla.thrift.example.entities.StudentThriftObject$StudentThriftObjectStandardScheme.read(StudentThriftObject.java:479)
    at com.vanilla.thrift.example.entities.StudentThriftObject.read(StudentThriftObject.java:413)
    at com.vanilla.thrift.controller.Main.main(Main.java:24)

我错过了什么吗?

1 个答案:

答案 0 :(得分:5)

你错过了Thrift的JSON与你的不同的事实。不写入字段名称,而是写入(和预期)指定的字段ID号。这是Thrift的JSON协议的一个例子:

[1,"MyService",2,1,{"1":{"rec":{"1":{"str":"Error: Process() failed"}}}}]

换句话说,Thrift不打算解析任何类型的JSON。它支持非常特定的JSON格式作为可能的传输之一。

但是,根据您的JSON数据的来源,Thrift可能仍然可以帮助您,如果您能够在双方使用它。在这种情况下,编写一个IDL来描述数据结构,将其提供给Thrift编译器,并将生成的代码和库的必要部分与您的项目集成。

如果JSON的来源超出您的范围,或者由于某种原因无法更改JSON格式,您需要找到另一种方式。

格式和语义是不同的野兽

在某种程度上,可以将整个问题与XML进行比较:有一种通用的XML语法,它告诉我们如何填充事物,以便任何符合标准的XML处理器都可以读取它们。

但是,如果我们从某人那里获得某个XML文件,那么了解XML的规则只是答案的一半。即使我们的XML解析器可以成功读取文件,因为它是格式良好的XML,我们需要知道数据的语义才能真正利用该文件中的内容:它是customer data record吗?或者它是SOAP envelope?也许是configuration file

这就是DTD或XML Schema发挥作用的地方,它们用于描述XML数据的内容。在不知道您丢失的逻辑结构的情况下,因为有无数种可能的方式来表达XML中的内容。除了JSON schema descriptions不太常用之外,JSON也是如此。

你的意思是,我们只需要告诉Thrift如何组织JSON?

不,因为Thrift背后的目的和理念是拥有一个框架来尽可能高效地解决/序列化事物和/或实现RPC服务器和客户端。它不打算具有通用文件解析器。相反,Thrift只读取和说出自己的一组格式,即plugged into the architecture as protocols:Thrift Binary,Thrift JSON,Thrift Compact等等。

你可以做什么:除了我在答案的第一部分中所说的,你可以考虑编写自己的自定义Thrift协议实现来支持你选择的特定JSON格式。这并不难,值得一试。