一文学习Thrift RPC

Thrift RPC引言

Thrift RPC的特点

Thrift 是一个RPC的框架,和Hessian RPC有什么区别,最重要的区别是Thrift可以做异构系统开发。
什么是异构系统,服务的提供者和服务的调用者是用不同语言开发的。

为什么会当前系统会有异构系统的调用?
做大数据一般使用Python,数据存储又要使用Hbase(Java),这就是异构系统了。
另外,还有遗留系统额整合,一个公司的系统,是随着业务增长的,在不同的业务可能由不同的编程语言开发的,想要完成一个共同的目标,就要不同系统之间进行交互。不同的系统要传输数据,首先要构建一个统一的格式,和编程语言无关。

设计一个异构系统的RPC,解决的核心问题是什么?
只要双方用各自的编程语言,网络编程建立连接,进行通信即可。
但是有一些挑战:
1.得需要精通不同编程语言的网络IO线程技术。
2.通信数据的格式,尤其是二进制的格式,不同编程语言的二进制数据量大小不一样。
所以我们引入了Thrift帮我们解决异构系统的调用。

ThriftRPC框架

  1. 基本概念:是apache组织开源的一个顶级异构系统RPC框架,用于完成异构系统的RPC通信。
    多种编程语言 Java C++ PHP Phyton Ruby Node.jsp…
    2007 FaceBook Thrift 开源。
  2. 特点:
    1. 跨语言支持
    2. 开发快速
    3. 学习简单 IDL语言
    4. 稳定
  3. Thrift的设计思想
    1. 针对于不同的编程语言提供了一个库(jar)
      作用:网络通信的代码 协议(序列化)相关的内容 java libthrift
    2. IDL语言 中立语言 用于服务发布
    3. Thrift命令把IDL语言 自动转换成 你需要的编程语言。
      在这里插入图片描述

Thrift 命令的安装

安装什么?

安装的是将IDL语言转换为相应的编程语言。

如何安装

mac brew install thrift
windows http://www.apache.org/dyn/closer.cgi?path=/thrift/0.18.0/thrift-0.18.0.exe
https://blog.51cto.com/u_15854865/5810927

如何验证: thrift -help thrift --version

针对于不同的编程语言 安装库

这里用Java

   <dependency><groupId>org.apache.thrift</groupId><artifactId>libthrift</artifactId><version>0.18.0</version></dependency>

IDL语法

IDEA 中 Thrift插件,插件目的提示 校验IDL语法。
IDL语法 必须 thrift

  • 注释

    #    单行注释
    //   单行注释
    /*
    *    多行注释 
    */
    
  • namespace

    namespace java com.liu
    namespace py com.liu指定编程语言,指定生成好的代码 包 
    
  • 基本类型

    1.  i8      有符号的8位整数   byte
    2.  i16     有符号的16位整数  short
    3.  i32     有符号的32位整数  int
    4.  i64     有符号的64位整数  long
    5.  double  64位浮点数       double
    6.  bool    布尔值          boolean
    7.  string  字符串 字符      "" ''  UTF-8
    
  • 集合类型

    list<T>   有序可重复    java.util.List 
    set<T>    无需不可重复  java.util.Set
    map<K,V>  k-v         java.util.Mapmap<i32,string> sex = {1:'female',2:'male'}
    list<i32> ages = [1,2,3,4]
    
  • struct 自定义对象

    struct User{1: string name ='sunshuai',2: optional i32 age,3: list<i32> ages = [1,2,3,4],4. required i32 hieght
    }1. struts 不能继承
    2. 成员与成员的分割,;
    3. 结构体里面的每一个字段 都要进行编号
    4. 变量类型 变量名
    5. optional 可选的 默认为每一个成员都加入的关键字代表这个字段在序列化过程中可选的。如果这个字段没有默认值,就不序列化,如果有默认值 就序列化.
    6. required 必选,有没有值都会序列化
    
  • 枚举 (enum)

    enum SEASON{SPRING = 1,SUMMERT = 2...
    }不支持嵌套,i32 
    
  • 异常 (Exception)

    exception MyException{1: i32 errorCode2: string message
    }
    
  • 服务 (Service)

    服务接口 
    service UserService{bool login(1:string name,2:string password)void register(1:User user) //User idl语言中的结构体 
    }注意:1. 异常service UserService{bool login(1:string name,2:string password) throws (1:MyException e,2:XXXException e)void register(1:User user) //User idl语言中的结构体 }2. oneway 表示客户端发起请求后不等待响应返回,只能和void 这种操作配合。service UserService{bool login(1:string name,2:string password) throws (1:MyException e,2:XXXException e)oneway void register(1:User user) //User idl语言中的结构体 }3. 继承service BaseService{void m1(1:string name)}service UserService extends BaseService{}
    
  • include

    作用:进行IDL模块化编程suns1.thriftstruct User{1:string name}
    suns2.thrift include "suns1.thrift"service UserService{void register(1:suns1.User user)}
    
  • Thrift把IDL生成对应代码的命令

    thrift --gen java xx.thrift 
    thrift --gen py xx.thriftthrift -r --gen java xx.thrift 
    

Thrift RPC的开发

环境搭建

1. 安装Thrift 作用:把IDL语言描述的接口内容,生成对应编程语言的代码,简化开发。
2. 引入依赖    作用:引入thrift针对于某一种编程语言的封装 (网络通信 协议【序列化】)<dependency><groupId>org.apache.thrift</groupId><artifactId>libthrift</artifactId><version>0.13.0</version></dependency><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-api</artifactId><version>1.7.32</version></dependency><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.2.9</version></dependency>

项目结构的定义

1. thrift-client  代表的是服务的调用者
2. thrift-server  代表的是服务的提供者
3. thrift-common  RPC编程共有的内容 1,实体类型 2,服务接口

Thrift核心对象

1. TTransport作用:底层封装网络通信TSocket 阻塞IO通信TNonblockdingTransport 非阻塞网络通信TFramedTransport 加入了封帧的操作 (压缩后 数据边界问题)  
2. TProtocol处理的协议 (序列化方式)TBinayProtocol 二进制进行序列化TCompactProtocol 压缩方式 处理二进制 TJSONProtocol  JSON进行序列化
3. TProcessor业务处理:把通信数据 和 业务功能整合在一起
4. TServer 服务端

thrift-common

1. 通过IDL语言 定义 client与服务端 共用的数据类型 和 服务接口 
2. client server端 引入 common模块

编写thrift文件:

namespace java com.liu
struct User{1:string name,2:string password
}service UserService{User queryUserByNamendPassword(1:string name,2:string password),void save(1:User user)
}

使用Thrift工具命令生成文件放在common模块中。
在这里插入图片描述

服务端

1. 实现服务接口 :idl语言生成的
2. 创建服务端代码

在这里插入图片描述
服务端实现之后,就要发布服务,如何发布服务?

public static void main1(String[] args) throws TTransportException {//TTransportTServerTransport tServerTransport =new TServerSocket(9002);//TBinaryProtocolTBinaryProtocol.Factory factory = new TBinaryProtocol.Factory();//TProcessorUserService.Processor processor = new UserService.Processor(new UserServiceImpl());TSimpleServer.Args tArgs = new TSimpleServer.Args(tServerTransport);tArgs.protocolFactory(factory);tArgs.processor(processor);//TServer发布服务TServer tServer =new TSimpleServer(tArgs);tServer.serve();}

客户端

客户端要向本地方法那样 调用远端方法。 代理
 public static void main(String[] args) throws TException {//传输方式 基于Socket阻塞 也可以使用非阻塞方式TTransport transport=new TSocket("localhost",9002);//传输协议 TBinaryProtocol 二进制格式 TCompactProtocol 压缩格式 TJSONProtocol json格式TProtocol tProtocol =new TBinaryProtocol(transport);UserService.Client userService = new UserService.Client(tProtocol);transport.open();User user = userService.queryUserByNamendPassword("liusir", "123");System.out.println("user = " + user);}

启动服务端之后,然后运行客户端可以看到服务成功了。
在这里插入图片描述

实战开发中的思考

在实战开发中,往往服务端的功能 其实已经开发完成。1. 现有本地的功能 服务端的功能 写好了。2. 根据系统的功能,才有可能决定 这个服务发布成 RPC。3. 所以直接在PRC方法里面调用我们已经写好的。
所以在发布RPC业务实现 IFace接口,主要通过原有的Service方法的调用,进行实现。这样维护性更好,也是实战中使用的方式。

TServer服务的相关内容

  • TServer类型

    1. 代表的是Thrift开发中的服务器。
    2. 功能:服务的开启serve()   服务的关闭stop()
    3. 阻塞 非阻塞 有没有线程池 Reactor模式TSimpleServer:        阻塞 单线程的服务器 (没有实战价值,只是用于测试)TThreadPoolServer:     阻塞 线程池的服务器 TNonBlockingServer:    非阻塞单线程的服务器 TThreadSelectorServer: 实现了主从版的Reactor(类似Netty)
    
  • 分析TSimpleServer

    目的:1. 了解Thrift相关类型的作用(源码的分析)2. 核实SimpleServer是一个阻塞的 单线程的服务器
    

111

  • TThreadPoolServer

    阻塞,引入了线程池 1. 如果使用,一定注意 默认的线程池 最大值 Integer.Max显然不成 需要maxWorkerThreads进行设置。
    2. 不能够让我们的线程复用,因为没有selector底层实现思路 把具体调用的Socket分配 WorkerProcess进行操作,而WorkerProcess从线程池中获得 线程资源。
    
  • TNonblockingServer

    底层连接 必须使用 TFreameTransport ,TCompactProtocol

    非阻塞  单线程Java NIO SocketChannel#configureBlocking
    ServerSocketChannel#configureBlockingselector
    
  • TThreadSelectorServer [主从版的Reactor模式的实现][实战中推荐]
    在这里插入图片描述

    • client

      public class TestClient1 {public static void main(String[] args) throws TException {//完成  与服务端 网络连接的连接TTransport tTransport = new TSocket("localhost", 9000);TFramedTransport tFramedTransport = new TFramedTransport(tTransport);tTransport.open();//创建协议TCompactProtocol tCompactProtocol = new TCompactProtocol(tFramedTransport);//创建代理  stub 存根 桩UserService.Client userService = new UserService.Client(tCompactProtocol);User user = userService.queryUserByNameAndPassword("xiaojr", "9090");System.out.println("user = " + user);}
      }
      
    • server

      public class TestServer1 {public static void main(String[] args) throws TTransportException {TNonblockingServerSocket tNonblockingServerSocket = new TNonblockingServerSocket(9000);TFramedTransport.Factory tFramedTransport = new TFramedTransport.Factory();TCompactProtocol.Factory factory = new TCompactProtocol.Factory();UserService.Processor processor = new UserService.Processor(new UserServiceImpl());TThreadedSelectorServer.Args arg = new TThreadedSelectorServer.Args(tNonblockingServerSocket);arg.transportFactory(tFramedTransport);arg.protocolFactory(factory);arg.processor(processor);TServer tServer = new TThreadedSelectorServer(arg);tServer.serve();}
      }
      
    • 1. client server thrift版本(maven) 一致
      2. client server通信的transport protocal 保持一致
      

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/247491.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

XPath判断当前选中节点的元素类型 Python lxml判断当前Element的元素类型 爬虫爬取页面分元素类型提取纯文本

背景&前言 不知道你们做爬虫的时候&#xff0c;有没有碰到和我一样的情况&#xff1a;将页面提取成纯文本的时候&#xff0c;由于页面中各种链接、加粗字体等&#xff0c;直接提取会造成结果一坨一坨的&#xff0c;非常不规整。有时候还要自己对标题等元素进行修改&#x…

14.java集合

文章目录 概念Collection 接口概念示例 Iterator 迭代器基本操作&#xff1a;并发修改异常增强循环遍历数组&#xff1a;遍历集合&#xff1a;遍历字符串&#xff1a;限制 list接口ListIteratorArrayList创建 ArrayList&#xff1a;添加元素&#xff1a;获取元素&#xff1a;修…

【Unity】粒子贴图异常白边问题

从PS制作的黑底&#xff0c;白光的贴图。放入Unity粒子中&#xff0c;拉远看会有很严重的白边&#xff0c;像马赛克一样。 材质使用&#xff1a;Mobile/Particles/Additive 经测试只使用一张黑色的图片&#xff0c;也会有白边。 解决方案&#xff1a; 关闭黑色底&#xf…

【UE 材质】闪电材质

效果 步骤 1. 新建一个材质这里命名为“M_Lighting” 打开“M_Lighting”&#xff0c;设置混合模式为半透明&#xff0c;着色模型为无光照 在材质图表中添加如下节点 其中&#xff0c;纹理采样节点的纹理是一个线条 此时预览窗口中效果如文章开头所示。

自然语言NLP学习

2-7 门控循环单元&#xff08;GRU&#xff09;_哔哩哔哩_bilibili GRU LSTM 双向RNN CNN 卷积神经网络 输入层 转化为向量表示 dropout ppl 标量 在物理学和数学中&#xff0c;标量&#xff08;Scalar&#xff09;是一个只有大小、没有方向的量。它只用一个数值就可以完全…

第十三章认识Ajax(四)

认识FormData对象 FormData对象用于创建一个表示HTML表单数据的键值对集合。 它可以用于发送AJAX请求或通过XMLHttpRequest发送表单数据。 以下是FormData对象的一些作用&#xff1a; 收集表单数据&#xff1a;通过将FormData对象与表单元素关联&#xff0c;可以方便地收集表…

AF647-羧酸,Alexa-Fluor 647-羧酸,适合用于标记蛋白质

您好&#xff0c;欢迎来到新研之家 文章关键词&#xff1a;AF647-carboxylic-acid &#xff0c;AF647-COOH&#xff0c;AF647-acid&#xff0c;Alexa-Fluor 647-acid&#xff0c;AF647-羧酸&#xff0c;Alexa-Fluor 647-羧酸 一、基本信息 产品简介&#xff1a;AF647&#x…

周报(20240128)

日期&#xff1a;2024.1.22 - 2024.1.28 本周工作&#xff1a; 1. 阅读论文 本周阅读了以下论文&#xff1a; 《BRAU-Net&#xff1a;用于医学图像分割的U形混合CNN-Transformer网络》 背景 精确的医学图像分割对于临床量化、疾病诊断、治疗计划和许多其他应用至关重要。基…

深度学习核心技术与实践之深度学习研究篇

非书中全部内容&#xff0c;只是写了些自认为有收获的部分。 Batch Normalization 向前传播 &#xff08;1&#xff09;三个主要任务&#xff1a;计算出每批训练数据的统计量。 对数据进行标准化 对标…

赛氪荣获“2023天津高新技术企业大会支持单位”

1月23日上午&#xff0c;2023天津市高新技术企业大会新闻发布会在天开高教科技园核心区综合服务中心召开&#xff0c;市高企协以及来自高校、企业、社会组织等80余人现场参会。 大会组委会秘书长张博航介绍到&#xff1a;“本次大会将实现自开办以来的多个首次&#xff0c;首次…

AIDL实践

先贴最后的文件目录&#xff1a; aidl/android/hardware/demo/IFoo.aidl&#xff1a; package android.hardware.demo;import android.hardware.demo.IFooCallback;VintfStability interface IFoo {void doFoo();int doFooWithParameter(int param);void registerCallback(IFo…

案例分析技巧-软件工程

一、考试情况 需求分析&#xff08;※※※※&#xff09;面向对象设计&#xff08;※※&#xff09; 二、结构化需求分析 数据流图 数据流图的平衡原则 数据流图的答题技巧 利用数据平衡原则&#xff0c;比如顶层图的输入输出应与0层图一致补充实体 人物角色&#xff1a;客户、…

力扣3. 无重复字符的最长子串(滑动窗口)

Problem: 3. 无重复字符的最长子串 文章目录 题目描述思路及解法复杂度Code 题目描述 思路及解法 由于题目要求求出字符串中最长的连续无重复字符的最长子串&#xff0c;所以利用这个特性我们可以比较容易的想到利用双指针中的滑动窗口技巧来解决&#xff0c;但在实际的求解中…

[机器学习]简单线性回归——梯度下降法

一.梯度下降法概念 2.代码实现 # 0. 引入依赖 import numpy as np import matplotlib.pyplot as plt# 1. 导入数据&#xff08;data.csv&#xff09; points np.genfromtxt(data.csv, delimiter,) points[0,0]# 提取points中的两列数据&#xff0c;分别作为x&#xff0c;y …

从CNN ,LSTM 到Transformer的综述

前情提要&#xff1a;文本大量参照了以下的博客&#xff0c;本文创作的初衷是为了分享博主自己的学习和理解。对于刚开始接触NLP的同学来说&#xff0c;可以结合唐宇迪老师的B站视频【【NLP精华版教程】强推&#xff01;不愧是的最完整的NLP教程和学习路线图从原理构成开始学&a…

TCP_拥塞控制

引言 24年春节马上就要到了&#xff0c;作为开车党&#xff0c;最大的期盼就是顺利回家过年不要堵车。梦想是美好的&#xff0c;但现实是骨感的&#xff0c;拥堵的道路让人苦不堪言。 在网络世界中&#xff0c;类似于堵车的问题也存在&#xff0c;而TCP&#xff08;Transmissi…

如何使用Everything随时随地远程访问本地电脑搜索文件

文章目录 前言1.软件安装完成后&#xff0c;打开Everything2.登录cpolar官网 设置空白数据隧道3.将空白数据隧道与本地Everything软件结合起来总结 前言 要搭建一个在线资料库&#xff0c;我们需要两个软件的支持&#xff0c;分别是cpolar&#xff08;用于搭建内网穿透数据隧道…

数据结构排序算详解(动态图+代码描述)

目录 1、直接插入排序&#xff08;升序&#xff09; 2、希尔排序&#xff08;升序&#xff09; 3、选择排序&#xff08;升序&#xff09; 方式一&#xff08;一个指针&#xff09; 方式二&#xff08;两个指针&#xff09; 4、堆排序&#xff08;升序&#xff09; 5、冒…

go包与依赖管理

包&#xff08;package&#xff09; 包介绍 Go语言中支持模块化的开发理念&#xff0c;在Go语言中使用包&#xff08;package&#xff09;来支持代码模块化和代码复用。一个包是由一个或多个Go源码文件&#xff08;.go结尾的文件&#xff09;组成&#xff0c;是一种高级的代码…

CSS之定位

定位在CSS当中是一个比较重要的点&#xff0c;接下来&#xff0c;让我为大家介绍一下定位吧&#xff01; 属性描述position-relative相对定位position-absolute绝对定位position-fixed固定定位position-sticky粘性定位position-static静态定位 一、相对定位 给元素设置 posi…