Thrift学习

Thrift源码剖析

Thrift源码分析及一个完整的例子

CSDN Thrift源码分析

Thrift二进制序列化TCompactProtocol与TBinaryProtocol的原理和区别

一个thrift的例子

thrift 分为四个模块

TransPort

将Socket包装成各种TransPort使用。

TIOStreamTransport和TSocket这两个类的结构对应着阻塞同步IO, TSocket封装了Socket接口

TNonblockingTrasnsort,TNonblockingSocket这两个类对应着非阻塞IO

TMemoryInputTransport封装了一个字节数组byte[]来做输入流的封装

TMemoryBuffer使用字节数组输出流ByteArrayOutputStream做输出流的封装

TFramedTransport则封装了TMemoryInputTransport做输入流,封装了TByteArryOutPutStream做输出流,作为内存读写缓冲区的一个封装。TFramedTransport的flush方法时,会先写4个字节的输出流的长度作为消息头,然后写消息体。和FrameBuffer的读消息对应起来。FrameBuffer对消息时,先读4个字节的长度,再读消息体

TFastFramedTransport是内存利用率更高的一个内存读写缓存区,它使用自动增长的byte,而不是每次都new一个byte[],提高了内存的使用率。其他和TFramedTransport一样,flush时也会写4个字节的消息头表示消息长度。

Protocol

协议和编解码是一个网络应用程序的核心问题之一,客户端和服务器通过约定的协议来传输消息(数据),通过特定的格式来编解码字节流,并转化成业务消息,提供给上层框架调用。

Thrift的协议比较简单,它把协议和编解码整合在了一起。抽象类TProtocol定义了协议和编解码的顶层接口。

抽象类 TProtocol,构造方法需要一个Transport类型的对象

方法分为两大类,write方法和read方法

方法名的结构:

(write/read) + (Struct,Message,File,Map,List,Set) + (Begin/End)

(write/read) + (Bool,I16,I32,I64,Double,String,Binary)

Processor

基类TProcessor,只有一个process(TProtocol in, TProtocol out)方法

TBaseProcessor继承自TProcessor:

 private final I iface;
 private final Map<String,ProcessFunction<I, ? extends TBase>> processMap;

实现process方法,去map找到对应的方法,并执行:

public boolean process(TProtocol in, TProtocol out) throws TException {
    TMessage msg = in.readMessageBegin();
    ProcessFunction fn = processMap.get(msg.name);
    fn.process(msg.seqid, in, out, iface);
    return true;
  }

TProcessorFactory有两个方法getProcessor(TTransport trans)isAsyncProcessor()两个方法。

Server

共有以下几种类型:

server的工作可以分为:监听socket链接、数据的读写和业务逻辑的处理三部分。

  • TSimpleServer

TSimpleServer的工作模式只有一个工作线程,循环监听新请求的到来并完成对请求的处理,它只是在简单的演示时候使用

  • TNonBlockingServer

TNonblockingServer工作模式,该模式也是单线程工作,但是该模式采用NIO的方式,所有的socket都被注册到selector中,在一个线程中通过seletor循环监控所有的socket,每次selector结束时,处理所有的处于就绪状态的socket,对于有数据到来的socket进行数据读取操作,对于有数据发送的socket则进行数据发送,对于监听socket则产生一个新业务socket并将其注册到selector中。

  • THsHaServer

THsHaServer类是TNonblockingServer类的子类,在5.2节中的TNonblockingServer模式中,采用一个线程来完成对所有socket的监听和业务处理,造成了效率的低下,THsHaServer模式的引入则是部分解决了这些问题。THsHaServer模式中,引入一个线程池来专门进行业务处理。但监听和读写数据还是主线程来做,因此可能有性能问题。

  • TThreadPoolServer

线程池模式中,数据读取和业务处理都交由线程池完成,主线程只负责监听新连接,因此在并发量较大时新连接也能够被及时接受。线程池模式比较适合服务器端能预知最多有多少个客户端并发的情况,这时每个请求都能被业务线程池及时处理,性能也非常高。

线程池模式的处理能力受限于线程池的工作能力,当并发请求数大于线程池中的线程数时,新请求也只能排队等待。

  • TThreadedSelectorServer

(1)一个AcceptThread线程对象,专门用于处理监听socket上的新连接;

(2) 若干个SelectorThread对象专门用于处理业务socket的网络I/O操作,所有网络数据的读写均是有这些线程来完成;

(3) 一个负载均衡器SelectorThreadLoadBalancer对象,主要用于AcceptThread线程接收到一个新socket连接请求时,决定将这个新连接请求分配给哪个SelectorThread线程。

(4) 一个ExecutorService类型的工作线程池,在SelectorThread线程中,监听到有业务socket中有调用请求过来,则将请求读取之后,交个ExecutorService线程池中的线程完成此次调用的具体执行;

FrameBuffer

FrameBuffer是ThriftNIO服务器端的一个核心组件,它一方面承担了NIO编程中的缓冲区的功能,另一方面还承担了RPC方法调用的职责。

方法调用

(1)自动生成的Iface接口,是远程方法的顶层接口

(2)自动生成的Processor类及相关父类,包括TProcessor接口,TBaseProcess抽象类

(3)ProcessFunction抽象类,抽象了一个具体的方法调用,包含了方法名信息,调用方法的抽象过程等

(4)TNonblcokingServer,是NIO服务器的默认实现,通过Args参数来配置Processor等信息

(5)FrameBuffer类,服务器NIO的缓冲区对象,这个对象在服务器端收到全包并解码后,会调用Processor去完成实际的方法调用

服务器端的方法的具体实现类,实现Iface接口

  • 通过IDL来定义接口: DemoService.thrift
namespace java com.thrift.test

service DemoService{
    string sayHi(1:string name);
}
  • 根据IDL自动生成代码
    thrift -r --gen java DemoService.thrift

  • 先生成一个IFace和IAsynceFace。

public interface Iface {

    public String sayHi(String name) throws org.apache.thrift.TException;

  }
  • 生成args对象

  • 生成一个client对象,包括发送请求和接收结果的方法。

public String sayHi(String name) throws org.apache.thrift.TException{
  send_sayHi(name);
  return recv_sayHi();
}

public void send_sayHi(String name) throws org.apache.thrift.TException{
  sayHi_args args = new sayHi_args();
  args.setName(name);
  sendBase("sayHi", args);
}

public String recv_sayHi() throws org.apache.thrift.TException
{
  sayHi_result result = new sayHi_result();
  receiveBase(result, "sayHi");
  if (result.isSetSuccess()) {
    return result.success;
  }
  throw new org.apache.thrift.TApplicationException(org.apache.thrift.TApplicationException.MISSING_RESULT, "sayHi failed: unknown result");
}

  • 生成processor对象
protected sayHi_args getEmptyArgsInstance() {
        return new sayHi_args();
      }

protected sayHi_result getResult(I iface, sayHi_args args) throws org.apache.thrift.TException {
        sayHi_result result = new sayHi_result();
        result.success = iface.sayHi(args.name);
        return result;
      }
  • 实现IFace接口
package com.thrift.test;

import org.apache.thrift.TException;

public class DemoServiceImpl implements DemoService.Iface{

    @Override
    public String sayHi(String name) throws TException {
        return "Hi " + name + ", from Thrift Server";
    }

}


  • Client

public class Client {
    public static void main(String[] args) throws Exception{
        TSocket socket = new TSocket("127.0.0.1", 9090);
        socket.setTimeout(3000);
        TTransport transport = new TFramedTransport(socket);
        TProtocol protocol = new TCompactProtocol(transport);
        transport.open();
        System.out.println("Connected to Thrfit Server");
        
        DemoService.Client client = new DemoService.Client.Factory()
                .getClient(protocol);
        String result = client.sayHi("ITer_ZC");
        System.out.println(result);
    }
}

  • server
public class Server {
 public static void main(String[] args){
     TNonblockingServerSocket socket;
     try {
         socket = new TNonblockingServerSocket(9090);
         TNonblockingServer.Args options = new TNonblockingServer.Args(socket);
         TProcessor processor = new DemoService.Processor<Iface>(new DemoServiceImpl());
         options.processor(processor);
         options.protocolFactory(new TCompactProtocol.Factory());
         TServer server = new TNonblockingServer(options);
         System.out.println("Thrift Server is running at 9090 port");
         server.serve();        
     } catch (Exception e) {
         throw new RuntimeException(e);
     }
 }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容