Golang RPC 之 Thrift

谢烟客 · 2017-03-17 05:52:34 · 18617 次点击 · 预计阅读时间 4 分钟 · 大约8小时之前 开始浏览    
这是一个创建于 2017-03-17 05:52:34 的文章,其中的信息可能已经有所发展或是发生改变。

Thrift 简介:

Thrift 是一款高性能、开源的 RPC 框架,产自 Facebook 后贡献给了 Apache,Thrift 囊括了整个 RPC 的上下游体系,自带序列化编译工具,因为 Thrift 采用的是二进制序列化,并且与 gRPC 一样使用的都是长连接建立 client 与 server 之间的通讯,相比于比传统的使用XML,JSON,SOAP等短连接的解决方案性能要快得多。
本篇只介绍 Golang 关于 Thrift 的基础使用。

安装

  • 安装 Thrift 的 Golang 库有两种方案:

    1. 直接通过 go get 命令安装,缺点是因为不可抗拒的网络因素大部分人可能会失败:$ go get git.apache.org/thrift.git/lib/go/thrift
    2. 通过源码安装:
      • 在 $GOPATH 的 src 目录下创建多层级目录:git.apache.org/thrift.git/lib/go
      • 从 github 上下载 thrift 0.10.0 的源码 ,解压后进入 thrift-0.10.0/lib/go 目录下,将 thrift 目录 copy 到刚创建的 $GOPATH/src/git.apache.org/thrift.git/lib/go 目录下
      • 在任意目录下执行 $ go install git.apache.org/thrift.git/lib/go/thrift 就完成了 golang 的 thrift 库的安装
  • 安装 Thrift 的 IDL 编译工具

    1. windows 平台下安装:

      直接下载:thrift complier 下载地址,下载完成后改名为:thrift.exe 并将其放入到系统环境变量下即可使用

    2. Linux 平台下安装:

      从 github 上下载 thrift 0.10.0 的源码,解压后进入:thrift-0.10.0/compiler/cpp 目录执行如下命令完成编译后,将其放入到系统环境变量下即可使用:
      $ mkdir cmake-build
      $ cd cmake-build
      $ cmake ..
      $ make

  • 验证是否安装成功:

    $ thrift -version,如果打印出来:Thrift version 0.10.0 表明 complier 安装成功

实践:

下面我们使用 Thrift 定义一个接口,该接口实现对传入的数据进行大写的格式化处理。

  • 创建 golang 项目 ThriftDemo 工程:

Paste_Image.png
  1. client目录下的 client.go 实现了客户端用于发送数据并打印接收到 server 端处理后的数据
  2. server 目录下的 server.go 实现了服务端用于接收客户端发送的数据,并对数据进行大写处理后返回给客户端
  3. thrift_file 用于存放 thrift 的 IDL 文件: *.thrift
  • 定义 Thrift RPC 接口

example.thrift:

namespace py example

struct Data {
    1: string text
}

service format_data {
    Data do_format(1:Data data),
}
  • 编译 thrift 文件

    进入 thrift_file 目录执行:$ thrift -out .. --gen go example.thrift,就会在 thrift_file 的同级目录下生成 golang 的包:example,其中 format_data-remote 是生成的测试代码可以不用特别关注


Paste_Image.png
  • 实现 server 端:
package main

import (
    "ThriftDemo/example"
    "strings"
    "git.apache.org/thrift.git/lib/go/thrift"
    "fmt"
    "log"
)

type FormatDataImpl struct {}

func (fdi *FormatDataImpl) DoFormat(data *example.Data) (r *example.Data, err error){
    var rData example.Data
    rData.Text = strings.ToUpper(data.Text)

    return &rData, nil
}

const (
    HOST = "localhost"
    PORT = "8080"
)

func main() {

    handler := &FormatDataImpl{}
    processor := example.NewFormatDataProcessor(handler)
    serverTransport, err := thrift.NewTServerSocket(HOST + ":" + PORT)
    if err != nil {
        log.Fatalln("Error:", err)
    }
    transportFactory := thrift.NewTFramedTransportFactory(thrift.NewTTransportFactory())
    protocolFactory := thrift.NewTBinaryProtocolFactoryDefault()

    server := thrift.NewTSimpleServer4(processor, serverTransport, transportFactory, protocolFactory)
    fmt.Println("Running at:", HOST + ":" + PORT)
    server.Serve()
}
  • 实现 client 端
package main

import (
    "git.apache.org/thrift.git/lib/go/thrift"
    "net"
    "fmt"
    "ThriftDemo/example"
    "log"
)

const (
    HOST = "localhost"
    PORT = "8080"
)

func main()  {
    tSocket, err := thrift.NewTSocket(net.JoinHostPort(HOST, PORT))
    if err != nil {
        log.Fatalln("tSocket error:", err)
    }
    transportFactory := thrift.NewTFramedTransportFactory(thrift.NewTTransportFactory())
    transport := transportFactory.GetTransport(tSocket)
    protocolFactory := thrift.NewTBinaryProtocolFactoryDefault()

    client := example.NewFormatDataClientFactory(transport, protocolFactory)

    if err := transport.Open(); err != nil {
        log.Fatalln("Error opening:", HOST + ":" + PORT)
    }
    defer transport.Close()


    data := example.Data{Text:"hello,world!"}
    d, err := client.DoFormat(&data)
    fmt.Println(d.Text)
}
  • 执行验证结果:
    1. 先启动 server,之后再执行 client
    2. client 侧控制台如果打印的结果为: HELLO,WORLD! ,证明 Thrift 的 RPC 接口定义成功

有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:谢烟客

查看原文:Golang RPC 之 Thrift

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

18617 次点击  
加入收藏 微博
6 回复  |  直到 2022-10-17 13:38:19
nickluo
nickluo · #1 · 7年之前

我用你的 example.thrift 的文件编译出来的 go example.go文件好多的错误,

yy1119
yy1119 · #2 · 7年之前

AL`AA$KV8X7{$%L)F1%~78X.png Hi,和你一样的代码,能请教一下原因吗

lmh860628
lmh860628 · #3 · 7年之前
yy1119yy1119 #2 回复

![AL`AA$KV8X7{$%L)F1%~78X.png](https://static.studygolang.com/180406/cc3e3d4742358e65bd440de2b8085799.png) Hi,和你一样的代码,能请教一下原因吗

这个要在main函数所在的文件运行。如果用goland一般的就不会,并且要先启动服务端

wangxu31
wangxu31 · #4 · 6年之前
lmh860628lmh860628 #3 回复

#2楼 @yy1119 这个要在main函数所在的文件运行。如果用goland一般的就不会,并且要先启动服务端

没看懂你这句回复呢?(这个要在main函数所在的文件运行。如果用goland一般的就不会,并且要先启动服务端) 编译出来的example.go很多错误,server和client也有一些调用函数参数不足的。

sbqqqf5
sbqqqf5 · #5 · 6年之前
wangxu31wangxu31 #4 回复

#3楼 @lmh860628 没看懂你这句回复呢?(这个要在main函数所在的文件运行。如果用goland一般的就不会,并且要先启动服务端) 编译出来的example.go很多错误,server和client也有一些调用函数参数不足的。

windows 平台,thrift 0.11 生成的 example 文件中有些错误,按照提示手动修改一下就好了。

oprot.Flush(ctx)

kay011
kay011 · #6 · 2年之前

不对

添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传