Thrift源码分析-Processor

[TOC]

TProcessor

public interface TProcessor {
  public boolean process(TProtocol in, TProtocol out) throws TException;
}

TProcessor定义了一个接口,负责从输入中获取请求信息,调用用户自己实现的服务,并将结果写入到输出中。

Thrift提供了两个实现了TProcessor接口的抽象类:TBaseProcessor & TBaseAsyncProcessor,本文只介绍TBaseProcessor。

ProcessFunction

在介绍TBaseProcessor之前,需要先介绍ProcessFunction抽象类,因为这个类和其子类是TBaseProcessor功能的实现者。

Thrift将IDL中定义的每个方法抽象为一个类,即ProcessFunction类。该类负责从输入中读取参数,调用用户服务(TBaseProcessor通过ProcessFunction完成对用户服务的调用),将响应写回到输出中。

其有两个主要方法:

public final void process(int seqid, TProtocol iprot, TProtocol oprot, I iface) throws TException {
    // 1. 读取参数
    args.read(iprot);
    // 2. 调用用户服务
    result = getResult(iface, args);
    // 3. 输出
    if(!isOneway()) { // 标记为oneway的请求不需要返回结果
      oprot.writeMessageBegin(new TMessage(getMethodName(), msgType, seqid));
      result.write(oprot);
      oprot.writeMessageEnd();
      oprot.getTransport().flush();
    }
}
// 在getResult中完成对用户服务的调用,具体调用什么方法由子类指定。
public abstract TBase getResult(I iface, T args) throws TException;

Thrift编译IDL文件生成java代码时,为各个方法自动生成相应的ProcessFunction子类。例如IDL文件中定义了如下方法:

void waveHand();

其对应的ProcessFunction子类如下所示:

public static class Processor<I extends Iface> extends org.apache.thrift.TBaseProcessor<I> implements org.apache.thrift.TProcessor {
    ...
    ...
    ...

    // 用户方法对应的ProcessFunction类定义在该用户服务对应的TBaseProcessor子类中,TBaseProcessor子类也是在编译IDL文件时由Thrift自动生成的
    public static class waveHand<I extends Iface> extends org.apache.thrift.ProcessFunction<I, waveHand_args> {
      public waveHand() {
        super("waveHand");
      }

      public waveHand_args getEmptyArgsInstance() {
        return new waveHand_args();
      }

      protected boolean isOneway() {
        return false;
      }

      @Override
      protected boolean handleRuntimeExceptions() {
        return false;
      }

      public waveHand_result getResult(I iface, waveHand_args args) throws org.apache.thrift.TException {
        waveHand_result result = new waveHand_result();
        
        // 调用用户服务
        iface.waveHand();
        return result;
      }
    }
    
    ...
    ...
    ...
  }

TBaseProcessor

讲完用户方法对应的ProcessFunction,在回过头来看 TBaseProcessor。其实TBaseProcessor和ProcessFunction的关系与用户定义在IDL中的service和方法的关系类似,这点在swift代码的实现中有更明显的表现。

TBaseProcessor维护了两个变量,一个是用户服务接口,一个是用户定义方法名与其对应的ProcessFunction子类的映射关系。从这里也可以看出,Thrift不支持方法的重载

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

TBaseProcessor实现了TProcessor接口,先从输入中获取方法名,然后找到相应的ProcessFunction实例,通过ProcessFunction实例完成对请求的处理。

@Override
public boolean process(TProtocol in, TProtocol out) throws TException {
// 获取方法名
TMessage msg = in.readMessageBegin();
// 获取相应的ProcessFunction实例
ProcessFunction fn = processMap.get(msg.name);
// 不存在则跑出方法不存在的异常
if (fn == null) {
  TProtocolUtil.skip(in, TType.STRUCT);
  in.readMessageEnd();
  TApplicationException x = new TApplicationException(TApplicationException.UNKNOWN_METHOD, "Invalid method name: '"+msg.name+"'");
  out.writeMessageBegin(new TMessage(msg.name, TMessageType.EXCEPTION, msg.seqid));
  x.write(out);
  out.writeMessageEnd();
  out.getTransport().flush();
  return true;
}
// 用ProcessFunction实例完成对请求的处理
fn.process(msg.seqid, in, out, iface);
return true;
}

Processor

在编译IDL文件的时候,Thrift为每个用户定义的服务生成一个TBaseProcessor的子类Processor。

Processor中包含各个用户定义方法对应的ProcessFunction子类,并在构造方法中完成对用户定义方法名与其对应的ProcessFunction子类实例的映射。

public Processor(I iface) {
  super(iface, getProcessMap(new java.util.HashMap<java.lang.String, org.apache.thrift.ProcessFunction<I, ? extends org.apache.thrift.TBase>>()));
}

protected Processor(I iface, java.util.Map<java.lang.String, org.apache.thrift.ProcessFunction<I, ? extends org.apache.thrift.TBase>> processMap) {
  super(iface, getProcessMap(processMap));
}

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

推荐阅读更多精彩内容