RPC算是近些年比较火热的概念了,随着微服务架构的兴起,RPC的应用越来越广泛。本文介绍了RPC和gRPC的相关概念,并且通过详细的代码示例介绍了gRPC的基本使用。

gRPC

RPC是什么

在分布式计算,远程过程调用(英语:Remote Procedure Call,缩写为 RPC)是一个计算机通信协议。该协议允许运行于一台计算机的程序调用另一个地址空间(通常为一个开放网络的一台计算机)的子程序,而程序员就像调用本地程序一样,无需额外地为这个交互作用编程(无需关注细节)。RPC是一种服务器-客户端(Client/Server)模式,经典实现是一个通过发送请求-接受回应进行信息交互的系统。

gRPC是什么

gRPC是一种现代化开源的高性能RPC框架,能够运行于任意环境之中。最初由谷歌进行开发。它使用HTTP/2作为传输协议。
在gRPC里,客户端可以像调用本地方法一样直接调用其他机器上的服务端应用程序的方法,帮助你更容易创建分布式应用程序和服务。与许多RPC系统一样,gRPC是基于定义一个服务,指定一个可以远程调用的带有参数和返回类型的的方法。在服务端程序中实现这个接口并且运行gRPC服务处理客户端调用。在客户端,有一个stub提供和服务端相同的方法。gRPC快速入门 - 图1

为什么要用gRPC

使用gRPC, 我们可以一次性的在一个.proto文件中定义服务并使用任何支持它的语言去实现客户端和服务端,反过来,它们可以应用在各种场景中,从Google的服务器到你自己的平板电脑—— gRPC帮你解决了不同语言及环境间通信的复杂性。使用protocol buffers还能获得其他好处,包括高效的序列号,简单的IDL以及容易进行接口更新。总之一句话,使用gRPC能让我们更容易编写跨语言的分布式代码。

安装gRPC

安装gRPC

  1. go get -u google.golang.org/grpc

安装Protocol Buffers v3

安装用于生成gRPC服务代码的协议编译器,最简单的方法是从下面的链接:https://github.com/google/protobuf/releases下载适合你平台的预编译好的二进制文件(protoc-<version>-<platform>.zip)。
下载完之后,执行下面的步骤:

  1. 解压下载好的文件
  2. protoc二进制文件的路径加到环境变量中

接下来执行下面的命令安装protoc的Go插件:

  1. go get -u github.com/golang/protobuf/protoc-gen-go

编译插件protoc-gen-go将会安装到$GOBIN,默认是$GOPATH/bin,它必须在你的$PATH中以便协议编译器protoc能够找到它。

安装指定

gRPC开发分三步

把大象放进冰箱分几步?

  1. 把冰箱门打开。
  2. 把大象放进去。
  3. 把冰箱门带上。

gRPC开发同样分三步:

  1. 编写.proto文件,生成指定语言源代码。
  2. 编写服务端代码
  3. 编写客户端代码

    gRPC入门示例

    编写proto代码

    gRPC是基于Protocol Buffers。
    Protocol Buffers是一种与语言无关,平台无关的可扩展机制,用于序列化结构化数据。使用Protocol Buffers可以一次定义结构化的数据,然后可以使用特殊生成的源代码轻松地在各种数据流中使用各种语言编写和读取结构化数据。
    关于Protocol Buffers的教程可以自行在网上搜索,本文默认读者熟悉Protocol Buffers
    1. syntax = "proto3"; // 版本声明,使用Protocol Buffers v3版本
    2. package pb; // 包名
    3. // 定义一个打招呼服务
    4. service Greeter {
    5. // SayHello 方法
    6. rpc SayHello (HelloRequest) returns (HelloReply) {}
    7. }
    8. // 包含人名的一个请求消息
    9. message HelloRequest {
    10. string name = 1;
    11. }
    12. // 包含问候语的响应消息
    13. message HelloReply {
    14. string message = 1;
    15. }
    执行下面的命令,生成go语言源代码:
    1. protoc -I helloworld/ helloworld/pb/helloworld.proto --go_out=plugins=grpc:helloworld
    gRPC_demo/helloworld/pb目录下会生成helloworld.pb.go文件。

    编写Server端Go代码

    1. package main
    2. import (
    3. "fmt"
    4. "net"
    5. pb "gRPC_demo/helloworld/pb"
    6. "golang.org/x/net/context"
    7. "google.golang.org/grpc"
    8. "google.golang.org/grpc/reflection"
    9. )
    10. type server struct{}
    11. func (s *server) SayHello(ctx context.Context, in *pb.HelloRequest) (*pb.HelloReply, error) {
    12. return &pb.HelloReply{Message: "Hello " + in.Name}, nil
    13. }
    14. func main() {
    15. // 监听本地的8972端口
    16. lis, err := net.Listen("tcp", ":8972")
    17. if err != nil {
    18. fmt.Printf("failed to listen: %v", err)
    19. return
    20. }
    21. s := grpc.NewServer() // 创建gRPC服务器
    22. pb.RegisterGreeterServer(s, &server{}) // 在gRPC服务端注册服务
    23. reflection.Register(s) //在给定的gRPC服务器上注册服务器反射服务
    24. // Serve方法在lis上接受传入连接,为每个连接创建一个ServerTransport和server的goroutine。
    25. // 该goroutine读取gRPC请求,然后调用已注册的处理程序来响应它们。
    26. err = s.Serve(lis)
    27. if err != nil {
    28. fmt.Printf("failed to serve: %v", err)
    29. return
    30. }
    31. }
    将上面的代码保存到gRPC_demo/helloworld/server/server.go文件中,编译并执行:
    1. cd helloworld/server
    2. go build
    3. ./server

    编写Client端Go代码

    1. package main
    2. import (
    3. "context"
    4. "fmt"
    5. pb "gRPC_demo/helloworld/pb"
    6. "google.golang.org/grpc"
    7. )
    8. func main() {
    9. // 连接服务器
    10. conn, err := grpc.Dial(":8972", grpc.WithInsecure())
    11. if err != nil {
    12. fmt.Printf("faild to connect: %v", err)
    13. }
    14. defer conn.Close()
    15. c := pb.NewGreeterClient(conn)
    16. // 调用服务端的SayHello
    17. r, err := c.SayHello(context.Background(), &pb.HelloRequest{Name: "q1mi"})
    18. if err != nil {
    19. fmt.Printf("could not greet: %v", err)
    20. }
    21. fmt.Printf("Greeting: %s !\n", r.Message)
    22. }
    将上面的代码保存到gRPC_demo/helloworld/client/client.go文件中,编译并执行:
    1. cd helloworld/client/
    2. go build
    3. ./client
    得到输出如下(注意要先启动server端再启动client端):
    1. $ ./client
    2. Greeting: Hello q1mi!
    此时我们的目录结构如下:
    1. ./gRPC_demo
    2. ├── go.mod
    3. ├── go.sum
    4. └── helloworld
    5. ├── client
    6. ├── client
    7. └── client.go
    8. ├── client.py
    9. ├── pb
    10. ├── helloworld.pb.go
    11. └── helloworld.proto
    12. └── server
    13. ├── server
    14. └── server.go

    gRPC跨语言调用

    接下来,我们演示一下如何使用gRPC实现跨语言的RPC调用。
    我们使用Python语言编写Client,然后向上面使用go语言编写的server发送RPC请求。

    生成Python代码

    gRPC_demo目录执行下面的命令:
    1. python -m grpc_tools.protoc -I helloworld/pb/ --python_out=helloworld/client/ --grpc_python_out=helloworld/client/ helloworld/pb/helloworld.proto
    上面的命令会在gRPC_demo/helloworld/client/目录生成如下两个python文件:
    1. helloworld_pb2.py
    2. helloworld_pb2_grpc.py

    编写Python版Client

    `gRPC_demo/helloworld/client/目录闯将client.py文件,其内容如下:
    1. # coding=utf-8
    2. import logging
    3. import grpc
    4. import helloworld_pb2
    5. import helloworld_pb2_grpc
    6. def run():
    7. # 注意(gRPC Python Team): .close()方法在channel上是可用的。
    8. # 并且应该在with语句不符合代码需求的情况下使用。
    9. with grpc.insecure_channel('localhost:8972') as channel:
    10. stub = helloworld_pb2_grpc.GreeterStub(channel)
    11. response = stub.SayHello(helloworld_pb2.HelloRequest(name='q1mi'))
    12. print("Greeter client received: {}!".format(response.message))
    13. if __name__ == '__main__':
    14. logging.basicConfig()
    15. run()
    将上面的代码保存执行,得到输出结果如下:
    1. gRPC_demo $ python helloworld/client/client.py
    2. Greeter client received: Hello q1mi!
    这里我们就实现了,使用python代码编写的client去调用Go语言版本的server了。
    点击右边的链接查看完整代码:gRPC_demo完整代码
    gRPC还有更多进阶用法,未完待续…