本文共 5812 字,大约阅读时间需要 19 分钟。
Table of Contents
Thrift是一种接口描述语言和二进制通讯协议,它被用来定义和创建跨语言的服务。它被当作一个(RPC)框架来使用,是由为“大规模跨语言服务开发”而开发的。它通过一个代码生成引擎联合了一个软件栈,来创建不同程度的、无缝的高效服务,可以使用、(基于兼容系统)、Cappuccino、、、、、、、、、、、、和。虽然它以前是由Facebook开发的,但它现在是的项目了。该实现被描述在2007年4月的一篇由Facebook发表的技术论文中,该论文现由Apache掌管。
https://www.jianshu.com/p/4723ce380b0e
Protobuf是一个语言中立、平台中立,对结构化数据进行序列化的可扩展机制。
我们在开发的时候开发了一个restful web service,就是基于rest的http调用,A系统作为客户端,B系统作为服务器端。A系统可以通过URL的方式携带一些数据去调用B所提供的接口然后返回相应的结果数据。这种方式我们也可以认为是RPC的一种实现方式。对于这种方式我们可以认为是平台独立的、语言独立的,也就是语言中立、平台中立。也就是我们可以用Python编写的客户端去调用Java编写的服务端,因为都是通过URL的方式调用。因为URL相当于契约,URL背后的代码调用者无需关心。
RPC框架调用基本模型:如person.getPersonByName(String name),首先客户端先序列化调用数据,传给服务端,服务端再反序列化提取调用信息,查询客户端所需要的数据,完成之后再序列化结果传回给客户端。客户端再反序列化得到结果。
Apache thrift是一个可伸缩的,并且跨语言的一种服务性的开发,他所完成的功能实际上和protobuf是类似的。简单来说,是Facebook公布的一款开源跨语言的RPC框架。
Thrift包含一套完整的栈来创建客户端和服务端程序。顶层部分是由Thrift定义生成的代码。而服务则由这个文件客户端和处理器代码生成。在生成的代码里会创建不同于内建类型的数据结构,并将其作为结果发送。协议和是的一部分。有了Thrift,就可以定义一个服务或改变通讯和传输协议,而无需重新编译代码。除了客户端部分之外,Thrift还包括服务器基础设施来集成协议和传输,如阻塞、非阻塞及多线程服务器。栈中作为I/O基础的部分对于不同的语言则有不同的实现。
Thrift支持众多通讯协议:
支持的传输协议有:
Thrift还提供众多的服务器,包括:
https://www.jianshu.com/p/4723ce380b0e
RPC全称为Remote Procedure Call,意为远程过程调用。
假设有两台服务器A,B.A服务器上部署着一个应用a,B服务器上部署着一个应用b,现在a希望能够调用b应用的某个函数(方法),但是二者不在同一个进程内,不能直接调用,就需要通过网络传输,在AB服务器之间建一条网络传输通道,a把参数传过去,b接收到参数调用自己的方法得到结果,再通过网络传回给a。
简单讲就是A通过网络来调用B的过程,这个过程要涉及的东西很多,比如多线程、Socket、序列化反序列化、网络I/O,很复杂。于是牛掰的程序员把这些封装起来做成一套框架供大家使用,就是RPC框架。
thrift通过一个中间语言IDL(接口定义语言)来定义RPC的数据类型和接口,这些内容写在以.thrift结尾的文件中,然后通过特殊的编译器来生成不同语言的代码,以满足不同需要的开发者。比如java开发者,就可以生成java代码,c++开发者可以生成c++代码,生成的代码中不但包含目标语言的接口定义、方法、数据类型,还包含有RPC协议层和传输层的实现代码。
Thrift是一种c/s的架构体系。TServer主要任务是高效的接受客户端请求,并将请求转发给Processor处理。
Thrift一些已经明确的优点包括:
Thrift由C++编写,但可以为众多语言创建代码。要创建一个Thrift服务,必须写一些Thrift文件来描述它,为目标语言生成代码,并且写一些代码来启动服务器及从客户端调用它。
Thrift将由这个描述信息生成独立的代码。例如,在Java里,PhoneType将是Phone类中一个简单的enum。
百度百科:
创建一个服务Hello,创建文件Hello.thrift,代码如下:
namespace java service.demoservice Hello{ string helloString(1:string para)}
终端进入Hello.thrift所在目录,执行命令:
thrift -r -gen java Hello.thrift
发现在当前目录下多了一个gen-java的目录,里面的有一个Hello.java的文件。这个java文件包含Hello服务的接口定义Hello.Iface,以及服务调用的底层通信细节,包括客户端的调用逻辑Hello.Client以及服务端的处理逻辑Hello.Processor。
创建一个Maven管理的Java项目,pom.xml中添加相关的依赖,并将Hello.java文件复制到项目中:
org.apache.thrift libthrift 0.10.0 org.slf4j slf4j-log4j12 1.7.5
创建HelloServiceImpl实现Hello.Iface接口:
package service.demo;import org.apache.thrift.TException;public class HelloServiceImpl implements Hello.Iface { public String helloString(String para) throws TException { return "result:"+para; }}
创建服务端实现代码HelloServiceServer,把HelloServiceImpl作为一个具体的处理器传递给Thrift服务器:
public class HelloServiceServer { /** * 启动thrift服务器 */ public static void main(String[] args) { try { System.out.println("服务端开启...."); // 1.创建TProcessor TProcessor tprocessor = new Hello.Processor(new HelloServiceImpl()); // 2.创建TserverTransport TServerSocket serverTransport = new TServerSocket(9898); // 3.创建TProtocol TBinaryProtocol.Factory factory = new TBinaryProtocol.Factory(); TServer.Args tArgs = new TServer.Args(serverTransport); tArgs.processor(tprocessor); tArgs.protocolFactory(factory); // 4.创建Tserver,传入需要的参数,server将以上内容集成在一起 TServer server = new TSimpleServer(tArgs); // 5.启动server server.serve(); }catch (TTransportException e) { e.printStackTrace(); } }}
创建客户端实现代码HelloServiceClient,调用Hello.client访问服务端的逻辑实现:
public class HelloServiceClient { public static void main(String[] args) { System.out.println("客户端启动...."); TTransport transport = null; try { transport = new TSocket("localhost", 9898, 30000); // 协议要和服务端一致 TProtocol protocol = new TBinaryProtocol(transport); Hello.Client client = new Hello.Client(protocol); transport.open(); String result = client.helloString("哈哈"); System.out.println(result); } catch (TTransportException e) { e.printStackTrace(); } catch (TException e) { e.printStackTrace(); } finally { if (null != transport) { transport.close(); } } }}
全部工作完成后,下面来测试一下,先执行服务端main方法,在执行客户端main方法,会在客户端控制台打印出:哈哈
。
转载地址:http://vlpaf.baihongyu.com/