Thrift之TProtocol类体系原理及源码详细解析之二进制协议类TBinaryProtocolT(TBinaryProtocol).doc_第1页
Thrift之TProtocol类体系原理及源码详细解析之二进制协议类TBinaryProtocolT(TBinaryProtocol).doc_第2页
Thrift之TProtocol类体系原理及源码详细解析之二进制协议类TBinaryProtocolT(TBinaryProtocol).doc_第3页
Thrift之TProtocol类体系原理及源码详细解析之二进制协议类TBinaryProtocolT(TBinaryProtocol).doc_第4页
Thrift之TProtocol类体系原理及源码详细解析之二进制协议类TBinaryProtocolT(TBinaryProtocol).doc_第5页
已阅读5页,还剩5页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

这个协议是Thrift支持的默认二进制协议,它以二进制的格式写所有的数据,基本上直接发送原始数据。因为它直接从TVirtualProtocol类继承,而且是一个模板类。它的模板参数就是一个封装具体传输发送的类,这个类才是真正实现数据传输的。这个类的定义上一节举例已经出现过了就不在列出来了。下面我就结合scribe的Log函数执行的具体过程来分析使用这个协议所执行的功能,看看二进制协议是怎样工作的。RPC调用使用到协议部分主要是在发送函数相关信息到服务器和接收服务器返回结果。现在我就结合Log函数的实现代码具体分析。首先看看Log函数的发送相关信息函数send_log(在文件scribe.cpp): 1 void scribeClient:send_Log(const std:vector & messages) 2 3 4 5 int32_t cseqid = 0; 6 7 oprot_-writeMessageBegin(Log, :apache:thrift:protocol:T_CALL, cseqid);/写入函数调用消息 8 9 scribe_Log_pargs args;10 11 args.messages = &messages;12 13 args.write(oprot_);/调用参数类自己的写入函数写入参数到服务器14 15 oprot_-writeMessageEnd();/写入消息调用写入16 17 oprot_-getTransport()-writeEnd();/结束传输层的写入18 19 oprot_-getTransport()-flush();/刷新传输流,让写入马上执行,因为RPC调用需要马上得到结果20 21 从上面代码可以看出:首先调用具体一个协议的writeMessageBegin函数,当然这个我们分析的是二进制协议,那就看看二进制协议这个函数的实现,代码如下: 1 template 2 3 uint32_t TBinaryProtocolT:writeMessageBegin(const std:string& name, 4 5 const TMessageType messageType, const int32_t seqid) 6 7 if (this-strict_write_) /判断是否需要强制写入版本号 8 9 int32_t version = (VERSION_1) | (int32_t)messageType);/本版号是协议号和消息类型的与结果10 11 uint32_t wsize = 0;/记录写入的长度12 13 wsize += writeI32(version);/写版本号14 15 wsize += writeString(name);/写消息名称,这就是函数名称Log16 17 wsize += writeI32(seqid);/写调用序列号18 19 return wsize;/返回写入的长度20 21 else 22 23 uint32_t wsize = 0;24 25 wsize += writeString(name);26 27 wsize += writeByte(int8_t)messageType);28 29 wsize += writeI32(seqid);30 31 return wsize;32 33 34 35 根据上面代码和注释可以看出,根据是否需要写入协议版本号写入的内有所差别,写入协议号的目的是可以坚持客户端和服务器端是否使用相同的协议来传输的数据,保证数据格式的正确性。二进制的协议定义如下:1 static const int32_t VERSION_MASK = 0xffff0000;/取得协议的掩码2 3 static const int32_t VERSION_1 = 0x80010000;/具体协议本版号具体写入又调用了自己实现的相应的数据类型写入函数,看看writeString是怎么实现的,如下: 1 template 2 3 uint32_t TBinaryProtocolT:writeString(const std:string& str) 4 5 uint32_t size = str.size();/取得字符串的长度(大小) 6 7 uint32_t result = writeI32(int32_t)size);/写入字符串的长度到服务器 8 9 if (size 0) 10 11 this-trans_-write(uint8_t*)str.data(), size);/调用具体某一个传输方式的写入函数写入字符串数据12 13 14 15 return result + size;/返回写入的大小16 17 从上面代码可以看出这些类型的函数就是将对应的数据类型写入服务器,而且具体写入在这里还没有真正的进行,因为后面会讲到的Transport相关类还会对传输方式进行包装。现在我们继续回到send_Log函数,写入函数调用的消息以后就开始写函数调用需要的参数,函数参数的写入是通过函数参数对应的封装类进行的,Log函数的参数封装类是scribe_Log_pargs,把对应的参数传递给这个类的对象,然后调用它自己的写入函数写入参数到服务器,代码如下: 1 uint32_t scribe_Log_pargs:write(:apache:thrift:protocol:TProtocol* oprot) const 2 3 uint32_t xfer = 0; 4 5 xfer += oprot-writeStructBegin(scribe_Log_pargs);/写入参数类的名称 6 7 xfer += oprot-writeFieldBegin(messages, :apache:thrift:protocol:T_LIST, 1);/写入字段名称和类型 8 9 10 11 /开始写入链表类型12 13 xfer += oprot-writeListBegin(:apache:thrift:protocol:T_STRUCT, (*(this-messages).size();14 15 std:vector :const_iterator _iter6;16 17 for (_iter6 = (*(this-messages).begin(); _iter6 != (*(this-messages).end(); +_iter6)18 19 20 21 xfer += (*_iter6).write(oprot);/依次写入链表参数类型里面的每一个22 23 24 25 xfer += oprot-writeListEnd();/结束链表类型写入26 27 28 29 xfer += oprot-writeFieldEnd();/写入字段结束30 31 xfer += oprot-writeFieldStop();/停止写入字段32 33 xfer += oprot-writeStructEnd();/写入参数结束34 35 return xfer;36 37 具体参数的写入函数根据参数的类型具体处理并写入到服务器端。这样整个函数调用就做完了,剩下的就是处理写入后的一些善后处理,看具体代码有注释。当函数调用的消息发送出去以后就开始准备接收函数远程调用的结果(异步调用除外),这里接收Log函数调用返回结果的函数是recv_log,代码如下: 1 ResultCode scribeClient:recv_Log() 2 3 4 5 int32_t rseqid = 0; 6 7 std:string fname; 8 9 :apache:thrift:protocol:TMessageType mtype;/接收返回消息的类型10 11 iprot_-readMessageBegin(fname, mtype, rseqid);/读取返回结果的消息12 13 if (mtype = :apache:thrift:protocol:T_EXCEPTION) /处理返回消息是异常的情况14 15 :apache:thrift:TApplicationException x;16 17 x.read(iprot_);/读取异常信息18 19 iprot_-readMessageEnd();20 21 iprot_-getTransport()-readEnd();22 23 throw x;/抛出异常信息24 25 26 27 if (mtype != :apache:thrift:protocol:T_REPLY) /处理不是正常回复的结果28 29 iprot_-skip(:apache:thrift:protocol:T_STRUCT);30 31 iprot_-readMessageEnd();32 33 iprot_-getTransport()-readEnd();34 35 36 37 if (pare(Log) != 0) /比较是否是Log函数调用返回的结果38 39 iprot_-skip(:apache:thrift:protocol:T_STRUCT);40 41 iprot_-readMessageEnd();42 43 iprot_-getTransport()-readEnd();44 45 46 47 ResultCode _return;48 49 scribe_Log_presult result;50 51 result.success = &_return;52 53 result.read(iprot_);/读取结果信息54 55 iprot_-readMessageEnd();56 57 iprot_-getTransport()-readEnd();58 59 if (result._isset.success) /成功就正常返回,否则抛出异常信息60 61 return _return;62 63 64 65 throw :apache:thrift:TApplicationException(:apache:thrift:TApplicationException:MISSING_RESULT, 66 67 Log failed: unknown result);/抛出不知道结果的异常信息,调用失败了68 69 接收RPC调用结果的函数都是根据返回消息的类型做相应处理,不成功就抛出相应的异常信息。首先这里调用二进制协议的readMessageBegin函数读取由二进制写入的消息(这个当然是服务器端写入的),这个函数代码实现如下: 1 template 2 3 uint32_t TBinaryProtocolT:readMessageBegin(std:string& name, 4 5 TMessageType& messageType, int32_t& seqid) 6 7 uint32_t result = 0; 8 9 int32_t sz;10 11 result += readI32(sz);/读取消息的头部(可能是协议版本号和消息类型的组合,也可能直接是消息)12 13 if (sz strict_read_) /要求读协议本版号,但是这种情况是不存在协议版本号的所以抛出异常34 35 throw TProtocolException(TProtocolException:BAD_VERSION,36 37 No version identifier. old protocol client in strict mode?);38 39 else 40 41 int8_t type;42 43 result += readStringBody(name, sz);/读取消息名称(也就是函数名称)44 45 result += readByte(type);/读取消息类型46 47 messageType = (TMessageType)type;48 49 result += readI32(seqid);/读取函数调用ID号50 51 52 53 54 55 return result;/f返回读取数据的长度56 57 上面的函数代码向我们展示了怎样处理基于二进制协议消息的读取和处理的过程,当然这个过程必须是建立在相应的写入消息的过程,只有按照相应的格式才能正确的处理。还有一点需要强调一下,就是每一种数据类型的写入和读取函数也是相对应的,在这里我没有具体分析每一个数据类型的写入函数了,其实也没有必要,也是这些代码都是很容易的,关键是读和写必须配合起来。到此一个完整的基于二进制协议的RPC调用分析完毕,下面对这个二进制协议进行一下简单的总结。(1)如果需要传输协议版本号,那么0-4字节就是协议版本号和消息类型;否则0-4字节就直接是消息名称(其实就是函数的名称)的长度,假设长度为len。(2)如果0-4字节是协议版本号和消息类型,那么5-8字节就是消息名称的长度,同样假设长度为len,然后再跟着len字节的消息名称;否则就是len字节的消息名称。(3)接下来如果没有带协议版本号的还有1字节的消息类型;(4)然后都是4字节的请求的序列号;(5)接着继续写入参数类型的结构体(但是二进制协议并没有真正写入,所以没有占用字节);(6)如果真正的有参数的话就继续一次为每一个参数写入1字节的参数类型(在前面已经给出了参数类型的定义,就是一个枚举)、2字节的参数序号和具体参数需要的长度;(7)具体参数长度的需求如下:a)对于以下具有固定长度的简单数据类型的参数:简单数据类型长度(字节)备注T_STOP=01T_VOID=1,1T_BOOL=21T_BYTE=31T_I08=31T_I16=62T_I32

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论