我创建了以下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)
我错过了什么吗?
答案 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格式。这并不难,值得一试。