[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 extends org.apache.thrift.TBaseProcessor implements org.apache.thrift.TProcessor {
...
...
...
// 用户方法对应的ProcessFunction类定义在该用户服务对应的TBaseProcessor子类中,TBaseProcessor子类也是在编译IDL文件时由Thrift自动生成的
public static class waveHand extends org.apache.thrift.ProcessFunction {
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> 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>()));
}
protected Processor(I iface, java.util.Map> processMap) {
super(iface, getProcessMap(processMap));
}
private static java.util.Map> getProcessMap(java.util.Map> processMap) {
processMap.put("sayHello", new sayHello());
processMap.put("waveHand", new waveHand());
processMap.put("findPerson", new findPerson());
processMap.put("showPerson", new showPerson());
return processMap;
}