rpc框架之 thrift 学习 2 - 基本概念
thrift的基本构架:

上图源自:http://jnb.ociweb.com/jnb/jnbJun2009.html
底层Underlying I/O以上的部分,都是由thrift编译器生成的代码,其中:
Your Code 这是根据thrift文件中定义的dto及service接口方法
FooService.Client及FooService.Processer是thrift生成的用于客户端及服务端的标准代码
Foo.read/write 参数对象及结果对象在传输时,最终需要在client、server间进行重写,红色框指的就是这个
TProtocal 指传输的内容是啥?(二进制?Json ? )由于TProtocal是一个抽象类,因此最终调用时,如果想从BinaryProtocal换成JsonProtocal,这部分代码也不用重新生成
TTransport 指用什么方式传输?(Scoket? Memory?File?)同样,TTransport是抽象类,运行时由具体子类决定运输方式
最底层的Underlying I/O则是依赖于各种语言的实现,负责底层的网络通讯,thrift最初是由c++写的,理论上讲,c++上的性能应该最好。
以上一章的demo为例如,QueryParameter及DemoService的类图如下:

点击图片可以查看大图,从类图上看,大量使用了内部类(inner class),对于dto对象,内部类基本上分为_Fileds, Schema,SchemaFactory 三类,
_Fields 是一个枚举,罗列了dto的各种属性成员,
Schema 封装了write/read方法
SchemaFactory 是一个工厂,用于创建Schema实例

服务接口的类图,就有点复杂了,密密麻麻象蜘蛛网,除了刚才的三大类外,DemoService的Inner Class中还有Client、Processor等类,大家有兴趣可以慢慢看。
TProtocal : 传输的内容(即:What? )

从类图上看,支持 压缩格式、二进制格式、Json格式 等。
TTransport : 传输的方式(即:How? )

Thrift支持的传输方式非常多,从类的命名就能大概看出一二。
TServer: Server的类图如下

基本上分为二大类:一类是同步阻塞的Server,一类是非阻塞模式的Server,其中THsHaServer是一个Half-Sync/Half-Async 半同步,半异步的server
meta_data 元数据

类图中的xxxMetaData,基本对应了 列表、K-V映射、(无重复元素)集合、结构(即:类)、枚举以及字段的元数据信息。
Schema :对不同类型的TProtocal的读写操作,在这里抽象出来。

Variable-Length Quantity VLQ 变长编码:
Thirft采用TCompactProtocol序列化时之所以高效,跟VLQ变长编码有很大关系,直接借下面这张图来说吧:

整数106903,在java中我们知道int占用4个bytes,也就是32bit,高位字节如果不满,用0填充(最高位符号位除外), 这样的话,很多用0填充的高位字节位置其实是浪费的,VLQ的基本思路是将2进制每7位分组,这样106903的2进制就可以分成3组,然后每1组的最高位设为1或0,如果为1,表示相邻的下一个字节还有内容,要继续读取,如果该位置为0,则表示结束了。
这样的话,106903最终只需要3个字节就可以存储了,节省了1个字节。
上述这一堆概念在运行时,是如何串起来的呢?
可以从TServer的部分源码中略知一二:
public abstract class TServer {
public static class Args extends AbstractServerArgs<Args> {
public Args(TServerTransport transport) {
super(transport);
}
}
public static abstract class AbstractServerArgs<T extends AbstractServerArgs<T>> {
final TServerTransport serverTransport;
TProcessorFactory processorFactory;
TTransportFactory inputTransportFactory = new TTransportFactory();
TTransportFactory outputTransportFactory = new TTransportFactory();
TProtocolFactory inputProtocolFactory = new TBinaryProtocol.Factory();
TProtocolFactory outputProtocolFactory = new TBinaryProtocol.Factory();
public AbstractServerArgs(TServerTransport transport) {
serverTransport = transport;
}
...
/**
* Core processor
*/
protected TProcessorFactory processorFactory_;
/**
* Server transport
*/
protected TServerTransport serverTransport_;
/**
* Input Transport Factory
*/
protected TTransportFactory inputTransportFactory_;
/**
* Output Transport Factory
*/
protected TTransportFactory outputTransportFactory_;
/**
* Input Protocol Factory
*/
protected TProtocolFactory inputProtocolFactory_;
/**
* Output Protocol Factory
*/
protected TProtocolFactory outputProtocolFactory_;
private boolean isServing;
protected TServerEventHandler eventHandler_;
// Flag for stopping the server
// Please see THRIFT-1795 for the usage of this flag
protected volatile boolean stopped_ = false;
protected TServer(AbstractServerArgs args) {
processorFactory_ = args.processorFactory;
serverTransport_ = args.serverTransport;
inputTransportFactory_ = args.inputTransportFactory;
outputTransportFactory_ = args.outputTransportFactory;
inputProtocolFactory_ = args.inputProtocolFactory;
outputProtocolFactory_ = args.outputProtocolFactory;
}
从上述源码可以看出,TServer 中包含有input/output二类TProtocol,即:体现了 数据进来和出去时传输的格式(Binary? Json?...),另外还有input与output的Transport,即:数据进来和出去的时候,如何传输?(Scoket? File?...),另外还有一个Processor,其子类是通过IDL(thrift定义文件)生成的,运行时必须传递进来具体的子类。
这样,传递什么数据(what)?用什么方式传输(how)? 以及数据如何处理(process)?都有了
而且从源码中,可以发现默认的input/output Protocol都是BinaryProtocol(14,15行)。
Server端启动的时序图如下:

Client调用的时序图:

注:上面这二张序列图均出自https://www.ibm.com/developerworks/cn/java/j-lo-apachethrift/
有了这些整体的概念后,上一篇中的示例,如果我们想换成TCompactProtocal,Client与Server的代码都得同步修改,这样二边才能一致:
Client端:
public class ThriftClient {
public static void main(String[] args) {
try {
TTransport transport;
transport = new TSocket("localhost", 9090);
transport.open();
//TProtocol protocol = new TBinaryProtocol(transport);
TProtocol protocol = new TCompactProtocol(transport);
DemoService.Client client = new DemoService.Client(protocol);
...
Server端:
public static void simple(DemoService.Processor processor) {
try {
TServerTransport serverTransport = new TServerSocket(9090);
Args args = new Args(serverTransport);
args.outputProtocolFactory(new TCompactProtocol.Factory());
args.inputProtocolFactory(new TCompactProtocol.Factory());
TServer server = new TSimpleServer(args.processor(processor));
//TServer server = new TSimpleServer(new Args(serverTransport).processor(processor));
System.out.println("Starting the simple server...");
server.serve();
} catch (Exception e) {
e.printStackTrace();
}
}
最后,对TCompactProtocol、TBinaryProtocol、TJSONProtocol这三种序列化协议简单测试下效率:
@Test
public void testProtocol() throws Exception {
QueryParameter parameter = getQueryParameter(); //CompactProtocol测试
TSerializer serializerCompact = new TSerializer(new TCompactProtocol.Factory());
byte[] bytes1 = serializerCompact.serialize(parameter);
System.out.println("CompactProtocol序列后的byte数组长度:" + bytes1.length); //BinaryProtocol测试
TSerializer serializerBinary = new TSerializer(new TBinaryProtocol.Factory());
byte[] bytes2 = serializerBinary.serialize(parameter);
System.out.println("BinaryProtocol序列后的byte数组长度:" + bytes2.length); //JsonProtocol测试
TSerializer serializerJson = new TSerializer(new TJSONProtocol.Factory());
byte[] bytes3 = serializerJson.serialize(parameter);
System.out.println("JsonProtocol序列后的byte数组长度:" + bytes3.length);
System.out.println(serializerJson.toString(parameter)); System.out.println("-----------");
TDeserializer deserializerCompact = new TDeserializer(new TCompactProtocol.Factory());
QueryParameter query1 = new QueryParameter();
deserializerCompact.deserialize(query1, bytes1);
System.out.println("CompactProtocol反序列化结果:" + query1.equals(parameter)); TDeserializer deserializerBinary = new TDeserializer(new TBinaryProtocol.Factory());
QueryParameter query2 = new QueryParameter();
deserializerBinary.deserialize(query2, bytes2);
System.out.println("BinaryProtocol反序列化结果:" + query2.equals(parameter)); TDeserializer deserializerJSON = new TDeserializer(new TJSONProtocol.Factory());
QueryParameter query3 = new QueryParameter();
deserializerJSON.deserialize(query3, bytes3);
System.out.println("JSONProtocol反序列化结果:" + query2.equals(parameter));
} private QueryParameter getQueryParameter(){
QueryParameter query = new QueryParameter();
short start = 1;
short end = 5;
query.setAgeStart(start);
query.setAgeEnd(end);
return query;
}
输出结果:
CompactProtocol序列后的byte数组长度:5
BinaryProtocol序列后的byte数组长度:11
JsonProtocol序列后的byte数组长度:29
{"1":{"i16":1},"2":{"i16":5}}
-----------
CompactProtocol反序列化结果:true
BinaryProtocol反序列化结果:true
JSONProtocol反序列化结果:true
TCompactProtocol优势明显,序列后的bytes长度只有JSON的1/5左右,可以大幅减少网络传输量。
参考文章:
http://dongxicheng.org/search-engine/thrift-rpc/
http://blog.chinaunix.net/uid-20357359-id-2876170.html
http://jnb.ociweb.com/jnb/jnbJun2009.html
https://www.ibm.com/developerworks/cn/java/j-lo-apachethrift/
http://www.cnblogs.com/brucewoo/tag/Thrift/
rpc框架之 thrift 学习 2 - 基本概念的更多相关文章
- rpc框架之 thrift 学习 1 - 安装 及 hello world
thrift是一个facebook开源的高效RPC框架,其主要特点是跨语言及二进制高效传输(当然,除了二进制,也支持json等常用序列化机制),官网地址:http://thrift.apache.or ...
- rpc框架之 thrift连接池实现
接前一篇rpc框架之HA/负载均衡构架设计 继续,写了一个简单的thrift 连接池: 先做点准备工作: package yjmyzz; public class ServerInfo { publi ...
- rpc框架之gRPC 学习 - hello world
grpc是google在github于2015年开源的一款RPC框架,虽然protobuf很早google就开源了,但是google一直没推出正式的开源框架,导致github上基于protobuf的r ...
- RPC框架之Thrift
目前流行的服务调用方式有很多种,例如基于SOAP消息格式的 Web Service,基于 JSON 消息格式的 RESTful 服务等.其中所用到的数据传输方式包括 XML,JSON 等,然而 XML ...
- RPC框架之Thrift分析(转)
一.简介 1.Thrift是Facebook开发的跨语言的RPC服务框架.随后贡献给Apache开源组织.成为RPC服务的主流框架. 2.特点: 优点: 跨语言,支持java.c/c ...
- rpc框架之 avro 学习 2 - 高效的序列化
同一类框架,后出现的总会吸收之前框架的优点,然后加以改进,avro在序列化方面相对thrift就是一个很好的例子.借用Apache Avro 与 Thrift 比较 一文中的几张图来说明一下,avro ...
- rpc框架之avro 学习 1 - hello world
avro是hadoop的一个子项目,提供的功能与thrift.Protocol Buffer类似,都支持二进制高效序列化,也自带RPC机制,但是avro使用起来更简单,无需象thrift那样生成目标语 ...
- 服务化实战之 dubbo、dubbox、motan、thrift、grpc等RPC框架比较及选型
转自: http://blog.csdn.net/liubenlong007/article/details/54692241 概述 前段时间项目要做服务化,所以我比较了现在流行的几大RPC框架的优缺 ...
- dubbo、dubbox、motan、thrift、grpc等RPC框架比较及选型
概述 前段时间项目要做服务化,所以我比较了现在流行的几大RPC框架的优缺点以及使用场景,最终结合本身项目的实际情况选择了使用dubbox作为rpc基础服务框架.下面就简单介绍一下RPC框架技术选型的过 ...
随机推荐
- 数据库 SQL语句小结(更新中)
################ Navicat,单条执行sql ################ Navicat,数据库管理工具, 在查询的页面有好多命令,若单条执行: 1:可选中要执行的一条sql ...
- 每日Scrum(5)
进入冲刺第五天,软件的界面设计成为主打,收集学校的很多美图是我们组的任务: 问题在于软件已很难有很大的改进,大方向也都是变不了的
- Linux umount设备时出现device is busy解决方法
在Linux中,有时使用umount命令去卸载LV或文件时,可能出现umount: xxx: device is busy的情况,如下案例所示 [root@DB-Server u06]# vgdisp ...
- 如何在开机时让Tomcat以进程的方式启动
一. 安装tomcat服务 1. 打开cmd命令窗口,进入到"tomcat安装路径/bin"目录下,运行"service.bat install"命令,安装to ...
- asp.net 导出Excel
分享一个asp.net 导出假Excel代码.优点,不用借助于任何插件比如(NPOI),复制代码,修改grid.DataSource直接导出. 先看导出后的效果图 System.Web.UI.WebC ...
- 烂泥:apache性能测试工具ab的应用
本文由秀依林枫提供友情赞助,首发于烂泥行天下. 网站性能压力测试是服务器网站性能调优过程中必不可缺少的一环.只有让服务器处在高压情况下,才能真正体现出软件.硬件等各种设置不当所暴露出的问题. 性能测试 ...
- 烂泥:php5.6源码安装及php-fpm配置与nginx集成
本文由秀依林枫提供友情赞助,首发于烂泥行天下. LNMP环境的搭建中,现在只有php没有源码安装过.这篇文章就把这个介绍下. 注意本篇文章使用的centos 6.5 64bit. 登陆centos下载 ...
- 六轴加速度传感器MPU6050官方DMP库到瑞萨RL78/G13的移植
2015年的电赛已经结束了.赛前接到器件清单的时候,看到带防护圈的多旋翼飞行器赫然在列,又给了一个瑞萨RL78/G13的MCU,于是自然联想到13年的电赛,觉得多半是拿RL78/G13做四旋翼的主控, ...
- spring为什么不能注入static变量
Spring 依赖注入 是依赖 set方法 set方法是 是普通的对象方法 static变量是类的属性 @Autowired private static JdbcTemplate jdbcTempl ...
- [转]Asp.net MVC使用Filter解除Session, Cookie等依赖
本文转自:http://www.cnblogs.com/JustRun1983/p/3279139.html 本文,介绍了Filter在MVC请求的生命周期中的作用和角色,以及Filter的一些常用应 ...