RPC框架之Thrift—实现Go和Java远程过程调用

简介: RPC框架之Thrift—实现Go和Java远程过程调用

1 概览

+-------------------------------------------+
| Server                                    |
| (single-threaded, event-driven etc)       |
+-------------------------------------------+
| Processor                                 |
| (compiler generated)                      |
+-------------------------------------------+
| Protocol                                  |
| (JSON, compact etc)                       |
+-------------------------------------------+
| Transport                                 |
| (raw TCP, HTTP etc)                       |
+-------------------------------------------+
复制代码

我们把他换成自己容易理解的方式:

网络异常,图片无法展示
|


2 分层解析

2.1 Transport(传输层)

传输层为网络的读写提供了一个简单的抽象。这使得Thrift能够将底层传输与系统的其他部分解耦(例如,序列化/反序列化)。

接口支持的方法:

  • open
  • close
  • read
  • write
  • flush

可传输的底层协议:

  • 文件传输(I/O)
  • HTTP

2.2 Protocol(协议层)

协议层抽象定义了一种机制来将内存中的数据结构映射到连线格式。换句话说,协议指定数据类型如何使用底层的Transport对自己进行编码/解码。因此,协议实现控制编码方案并负责(反)序列化。这种意义上的协议的一些例子包括JSON、XML、纯文本、压缩二进制文件等。

接口支持的方法:

写(编码)方法:

  • writeMessageBegin(name, type, seq)
  • writeMessageEnd()
  • writeStructBegin(name)
  • writeStructEnd()
  • writeFieldBegin(name, type, id)
  • writeFieldEnd()
  • writeFieldStop()
  • writeMapBegin(ktype, vtype, size)
  • writeMapEnd()
  • writeListBegin(etype, size)
  • writeListEnd()
  • writeSetBegin(etype, size)
  • writeSetEnd()
  • writeBool(bool)
  • writeByte(byte)
  • writeI16(i16)
  • writeI32(i32)
  • writeI64(i64)
  • writeDouble(double)
  • writeString(string)

读(解码)方法:

  • name, type, seq = readMessageBegin()
    readMessageEnd()
  • name = readStructBegin()
    readStructEnd()
  • name, type, id = readFieldBegin()
    readFieldEnd()
  • k, v, size = readMapBegin()
    readMapEnd()
  • etype, size = readListBegin()
    readListEnd()
  • etype, size = readSetBegin()
    readSetEnd()
  • bool = readBool()
  • byte = readByte()
  • i16 = readI16()
  • i32 = readI32()
  • i64 = readI64()
  • double = readDouble()
  • string = readString()

支持的编码协议:

  • 二进制
  • compact
  • json

2.3 Processor(处理器层)

处理器封装了从输入流读取数据和向输出流写入数据的能力。输入和输出流由Protocol对象表示。

接口方法样例:

interface TProcessor {
    bool process(TProtocol in, TProtocol out) throws TException
}
复制代码

2.4 Server(应用服务层)

一个服务器集合了上面描述的所有不同的特性:

  • 创建一个运输
  • 为传输创建输入/输出协议
  • 创建基于输入/输出协议的处理器
  • 等待传入的连接,并将它们交给处理器

3 thrift安装

下载页:thrift.apache.org/download

下载Windows版本:

网络异常,图片无法展示
|


下载后可以直接使用,或者放在固定文件设置环境变量

网络异常,图片无法展示
|


4 Go使用thrift

4.1 项目结构和依赖安装

网络异常,图片无法展示
|


依赖安装:

go get -u github.com/apache/thrift
复制代码

4.2 编写thrift文件并编译

namespace go hello
struct HelloReq {
    1: string msg;
}
struct HelloResp {
    1: string msg;
}
service HelloService {
    //返回值类型 方法名(参数序号:参数类型 参数名);
    HelloResp Hello(1: HelloReq req);
}
复制代码

编译:

thrift -r --gen go hello.thrift
复制代码

4.3 服务端代码

type HelloServiceImpl struct {
}
func (e *HelloServiceImpl) Hello(ctx context.Context, req *hello.HelloReq) (*hello.HelloResp, error) {
   fmt.Printf("message from client: %v\n", req.GetMsg())
   res := &hello.HelloResp{
      Msg: "Hi Client ~",
   }
   return res, nil
}
func main() {
  transport, err := thrift.NewTServerSocket(":9898")
  if err != nil {
    panic(err)
  }
  handler := &HelloServiceImpl{}
  processor := hello.NewHelloServiceProcessor(handler)
  //也可以选择Framed传输协议,但是必须确保服务端和客户端的传输协议一致
  transportFactory := thrift.NewTBufferedTransportFactory(1024)
  //应对Java客户端
  //factory := thrift.NewTFramedTransportFactory(thrift.NewTTransportFactory())
  //可以选择任意一种编码协议,但是必须确保服务端和客户端的编码协议一致
  protocolFactory := thrift.NewTBinaryProtocolFactoryConf(&thrift.TConfiguration{}) //布尔参数strictRead, strictWrite,读和写时是否加入版本校验。
  server := thrift.NewTSimpleServer4(
    processor,
    transport,
    transportFactory,
    //factory,
    protocolFactory,
  )
  if err := server.Serve(); err != nil {
    panic(err)
  }
}
复制代码

4.4 客户端代码

func main() {
  transportFactory := thrift.NewTBufferedTransportFactory(1024)
  //可以选择任意一种通信协议,但是必须确保服务端和客户端的通信协议一致
  protocolFactory := thrift.NewTBinaryProtocolFactoryConf(&thrift.TConfiguration{}) //布尔参数strictRead, strictWrite,读和写时是否加入版本校验。
  //Java服务端
  //factory := thrift.NewTCompactProtocolFactoryConf(&thrift.TConfiguration{})
  transport:= thrift.NewTSocketConf("127.0.0.1:9898",&thrift.TConfiguration{})
  useTransport, err := transportFactory.GetTransport(transport)
  client := hello.NewHelloServiceClientFactory(useTransport, protocolFactory)
  if err := transport.Open(); err != nil {
    fmt.Fprintln(os.Stderr, "Error opening socket to 127.0.0.1:9898", " ", err)
    os.Exit(1)
  }
  defer transport.Close()
  req := &hello.HelloReq{Msg: "Hello Server ~"}
  res, err := client.Hello(context.Background(), req)
  if err != nil {
    log.Println("Echo failed:", err)
    return
  }
  log.Println("response:", res.Msg)
}
复制代码

4.5 验证

网络异常,图片无法展示
|


5 Java使用thrift

5.1 项目结构和依赖

网络异常,图片无法展示
|


依赖:

<dependency>
    <groupId>org.apache.thrift</groupId>
    <artifactId>libthrift</artifactId>
    <version>0.15.0</version>
</dependency>
复制代码

5.2 编写thrift文件并编译

namespace java hello
struct HelloReq {
    1: string msg;
}
struct HelloResp {
    1: string msg;
}
service HelloService {
    HelloResp Hello(1: HelloReq req);
}
复制代码

编译:

thrift -r --gen java hello.thrift
复制代码

注意:编译完成之后需要将生成的.java文件放到项目结构中的对应位置。

5.3 服务端代码

public class ThriftServer {
    public static void main(String[] args) throws TTransportException {
        //设置服务器端口  TNonblockingServerSocket-非堵塞服务模型
        TNonblockingServerSocket serverSocket = new TNonblockingServerSocket(8899);
        //参数设置
        THsHaServer.Args arg = new THsHaServer.Args(serverSocket).minWorkerThreads(2).maxWorkerThreads(4);
        //处理器
        //PersonService.Processor<PersonServiceImpl> processor = new PersonService.Processor<>(new PersonServiceImpl());
        HelloService.Processor<HelloServiceImpl> processor = new HelloService.Processor<>(new HelloServiceImpl());
        arg.protocolFactory(new TCompactProtocol.Factory());
        arg.transportFactory(new TFramedTransport.Factory());
        arg.processorFactory(new TProcessorFactory(processor));
        TServer server = new THsHaServer(arg);
        System.out.println("Thrift 服务端启动成功");
        server.serve();
    }
}
//Handler
class HelloServiceImpl implements HelloService.Iface {
    @Override
    public HelloResp Hello(HelloReq req) throws TException {
        System.out.println(req.msg);
        return new HelloResp("Hello Client ~");
    }
}
复制代码

5.4 客户端代码

public class ThriftClient {
    public static void main(String[] args) throws TException {
        TTransport transport = new TFramedTransport(new TSocket("localhost", 8899), 600);
        TProtocol protocol = new TCompactProtocol(transport);
        HelloService.Client client = new HelloService.Client(protocol);
        transport.open();
        String msg = client.Hello(new HelloReq("Hi Server ~")).msg;
        System.out.println(msg);
        transport.close();
    }
}
复制代码

6 使用thrift进行Java和Go的远程通信

  • 原则:客户端与服务端的协议一致即可

例如Java做客户端:

public class ThriftClientGo {
    public static void main(String[] args) throws TTransportException {
        TTransport transport = new TFramedTransport(new TSocket("127.0.0.1", 9898), 8000);
        TBinaryProtocol binaryProtocol = new TBinaryProtocol(transport);
        HelloService.Client client = new HelloService.Client(binaryProtocol);
        try {
            transport.open();
            String msg = client.Hello(new HelloReq("Hi Server~")).msg;
            System.out.println(msg);
        }catch (Exception ex){
            throw new RuntimeException(ex.getMessage(),ex);
        }finally {
            transport.close();
        }
    }
}
复制代码

参考文章:

thrift.apache.org/docs/concep…

thrift.apache.org/docs/concep…

blog.csdn.net/pengpengzho…



相关文章
|
2月前
|
自然语言处理 负载均衡 API
gRPC 一种现代、开源、高性能的远程过程调用 (RPC) 可以在任何地方运行的框架
gRPC 是一种现代开源高性能远程过程调用(RPC)框架,支持多种编程语言,可在任何环境中运行。它通过高效的连接方式,支持负载平衡、跟踪、健康检查和身份验证,适用于微服务架构、移动设备和浏览器客户端连接后端服务等场景。gRPC 使用 Protocol Buffers 作为接口定义语言,支持四种服务方法:一元 RPC、服务器流式处理、客户端流式处理和双向流式处理。
|
5月前
|
Dubbo 网络协议 Java
RPC框架:一文带你搞懂RPC
这篇文章全面介绍了RPC(远程过程调用)的概念、原理和应用场景,解释了RPC如何工作以及为什么在分布式系统中广泛使用,并探讨了几种常用的RPC框架如Thrift、gRPC、Dubbo和Spring Cloud,同时详细阐述了RPC调用流程和实现透明化远程服务调用的关键技术,包括动态代理和消息的编码解码过程。
RPC框架:一文带你搞懂RPC
|
5月前
|
安全 Java Go
Java&Go泛型对比
总的来说,Java和Go在泛型的实现和使用上各有特点,Java的泛型更注重于类型安全和兼容性,而Go的泛型在保持类型安全的同时,提供了更灵活的类型参数和类型集的概念,同时避免了运行时的性能开销。开发者在使用时可以根据自己的需求和语言特性来选择使用哪种语言的泛型特性。
60 7
|
4月前
|
XML 负载均衡 监控
分布式-dubbo-简易版的RPC框架
分布式-dubbo-简易版的RPC框架
|
5月前
|
XML 存储 JSON
(十二)探索高性能通信与RPC框架基石:Json、ProtoBuf、Hessian序列化详解
如今这个分布式风靡的时代,网络通信技术,是每位技术人员必须掌握的技能,因为无论是哪种分布式技术,都离不开心跳、选举、节点感知、数据同步……等机制,而究其根本,这些技术的本质都是网络间的数据交互。正因如此,想要构建一个高性能的分布式组件/系统,不得不思考一个问题:怎么才能让数据传输的速度更快?
125 1
|
6月前
|
分布式计算 负载均衡 数据安全/隐私保护
什么是RPC?有哪些RPC框架?
RPC(Remote Procedure Call,远程过程调用)是一种允许运行在一台计算机上的程序调用另一台计算机上子程序的技术。这种技术屏蔽了底层的网络通信细节,使得程序间的远程通信如同本地调用一样简单。RPC机制使得开发者能够构建分布式计算系统,其中不同的组件可以分布在不同的计算机上,但它们之间可以像在同一台机器上一样相互调用。
171 8
|
6月前
|
网络协议 Dubbo Java
什么是RPC?RPC和HTTP对比?RPC有什么缺点?市面上常用的RPC框架?
选择合适的RPC框架和通信协议,对于构建高效、稳定的分布式系统至关重要。开发者需要根据自己的业务需求和系统架构,综合考虑各种因素,做出适宜的技术选型。
521 1
|
5月前
|
Rust JavaScript Java
简单对比Java、Python、Go、Rust等常见语言计算斐波拉契数的性能
简单对比Java、Python、Go、Rust等常见语言计算斐波拉契数的性能
|
6月前
|
负载均衡 Java
使用Java实现RPC框架
使用Java实现RPC框架
|
6月前
|
负载均衡 Java