Java中使用Thrift

Java中使用Thrift

Thrift建立(server端)

定义一个thrift接口:

java 复制代码
service ThriftService {
    void myFunc();
}

通过 thrift compiler 生成 java 代码

java 复制代码
thrift --gen java ThriftService.thrift

生成了Thrift的接口类:

java 复制代码
public class ThriftService {
	// 同步和异步调用接口
    public interface Iface {...}
    public interface AsyncIface {...}
    
    // 接口的实现类(同步和异步)
    public static class Client {...}
    public static class AsyncClient {...}   
}

实现server端:

java 复制代码
public class Server {
    public static class ThriftServiceImpl implements ThriftService.Iface{
 
        @Override
        public String myFunc(String name) throws TException {
            return "My Thrift Function";
        }
    }
 
    public static void main(String[] args) throws TTransportException {
        // 获取实现
        ThriftServiceImpl impl= new ThriftServiceImpl();
        // 接口与实现类的绑定关系在这里完成
        HelloWorldService.Processor<ThriftServiceImpl> processor = new ThriftService.Processor<ThriftServiceImpl>(impl);
        // 构建服务器
        TServerTransport serverTransport = new TServerSocket(9090);
        TServer server = new TSimpleServer(new Args(serverTransport).processor(processor));
        server.serve();
    }
}

Thrift调用(client端)

client端可以直接调用接口,二不用关心具体的实现类。

1 直接建立thrift通信

java 复制代码
public class Client {
    public static void main(String[] args) throws TException {
        TTransport transport = new TSocket("localhost", 9090);
        transport.open();
        TProtocol protocol = new TBinaryProtocol(transport);
        ThriftService.Client client = new ThriftService.Client(protocol);
		client.myFunc();
        transport.close();
    }
}

或者:

java 复制代码
	TFastFramedTransport transport = new TFastFramedTransport(new TSocket(host, port));
	TBinaryProtocol protocol = new TBinaryProtocol(transport);
	SimpleSerice.Client client = new SimpleService.Client(protocol);
	try {
	    transport.open();
	    client.myFunc();
	} finally {
	    transport.close();
	}

通信层协议,使用TTransport接口来描述,Thrift 提供了几种 transport 类供使用:

  • TSocket:BIO
  • TFramedTransport:NIO
  • TMemoryTransport:内存I/O

需要注意的是,transport 客户端要和服务端对应,比如如果要对 NIO 模型的服务端进行调研,transport 就必须使用 TFramedTransport或TFastFramedTransport。

传输层协议,引用 transport 并提供 thrift 的序列化协议,Thrift 也提供了几种常用的传输协议:

  • TBinaryProtocol:二进制协议
  • TCompactProtocol:压缩格式
  • TJSONProtocol:JSON 协议

2 使用JDK动态代理

首先创建代理类,这里只需要引用 thrift 接口中的 Service 类,并通过引用发现内部 Client 类:

java 复制代码
public class ThriftClientProxy implements InvocationHandler {

    private Class clazz;
    private Class clientClazz;
    private String host;
    private int port;

    public Object newInstance(Class clazz, String host, int port) {
        this.clazz = clazz;
        this.host = host;
        this.port = port;
        this.clientClazz = Class.forName(classs.getName() + "$Client");

        return Proxy.newProxyInstance(
            getClass().getClassLoader(),
            clientClazz.getInterfaces(),
            this
        );
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        TFastFramedTransport transport = new TFastFramedTransport(new TSocket(host, port));
        TBinaryProtocol protocol = new TBinaryProtocol(transport);
        Class[] classes = {TProtocol.class};
        try {
            transport.open();
            return method.invoke(clientClazz.getConstructor(classes).newInstance(protocol), args);
        }
        finally {
            transport.close();
        }
    }
}

之后将相应的 Iface 注册到 IOC 容器中即可。

java 复制代码
@Configuration
public class ThriftClientConfiguration {

    @Value("${simple.service.host}")
    private String host;

    @Value("${simple.service.port}")
    private int port;

    @Bean
    public SimpleService.Iface thriftClient() {
        return (SimpleService.Iface) new ThriftClientProxy().newInstance(SimpleService.class, host, port);
    }
}

业务代码中只需要注入 SimpleService.Iface,并直接调用接口方法,而完全不需要关注内部逻辑实现。

相关推荐
Swift社区33 分钟前
在 Swift 中实现字符串分割问题:以字典中的单词构造句子
开发语言·ios·swift
没头脑的ht34 分钟前
Swift内存访问冲突
开发语言·ios·swift
没头脑的ht37 分钟前
Swift闭包的本质
开发语言·ios·swift
wjs202440 分钟前
Swift 数组
开发语言
吾日三省吾码2 小时前
JVM 性能调优
java
stm 学习ing2 小时前
FPGA 第十讲 避免latch的产生
c语言·开发语言·单片机·嵌入式硬件·fpga开发·fpga
湫ccc3 小时前
《Python基础》之字符串格式化输出
开发语言·python
弗拉唐3 小时前
springBoot,mp,ssm整合案例
java·spring boot·mybatis
oi773 小时前
使用itextpdf进行pdf模版填充中文文本时部分字不显示问题
java·服务器