Protobuf&GRPC 的基本使用

简介: 首先对比一下Java自带的序列化与Protobuf的序列化的大小差异public static void main(String[] args) throws Exception { Person.
  • 首先对比一下Java自带的序列化与Protobuf的序列化的大小差异

    public static void main(String[] args) throws Exception {
        Person.User person = Person.User.newBuilder().setName("wangziqiang")
                .setAge(20)
                .setAddress("hebei")
                .setGender("男")
                .setTimestamp(System.currentTimeMillis()).build();
        ObjectOutputStream os = new ObjectOutputStream(new FileOutputStream("javaobj"));
        os.writeObject(person);
        os.close();
    
        person.writeTo(new FileOutputStream("protoobj"));
    }
  • 之后的javaobj是217字节,而proto是34字节,很明显的大小差异,下面是proto的反序列化代码

    public void test() throws Exception {
        Person.User user = Person.User.parseFrom(new FileInputStream("protoobj"));
        System.out.println(user);
    }
    /**
     * name: "wangziqiang"
     * age: 20
     * address: "hebei"
     * gender: "\347\224\267"
     * timestamp: 1546783663870
     */
  • 下面说一下序列化和反序列化的速度,直接上图,图来自网上

markdown_img_paste_20190106221200163

markdown_img_paste_20190106221208920

  • 另附一篇protobuf的测试帖子: 即时通讯网,看完后只有一个结论:使用protobuf基本上是稳赚不亏的
  • 之前自己接触过AVRO,也是可以进行序列化和RPC的框架,然后自己就产生了一个疑问:protobuf可以直接做RPC吗?网上的帖子大多是C的,自己就产生了一些误导,认为protobuf可以直接来做RPC的应用,但是这是错误的,经过请教,在钉钉群和云栖社区的问答都给了我很好的答案,如果你也有这个疑问,请移步到 我的云栖问答-protobuf
  • 由于gRPC也是Google开源的RPC框架,所以我将使用这个框架做一个小demo,本质上是跟着网上贴的代码,先会用再说
  • maven

    <dependencies>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-netty</artifactId>
            <version>1.17.1</version>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-protobuf</artifactId>
            <version>1.17.1</version>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-stub</artifactId>
            <version>1.17.1</version>
        </dependency>
    </dependencies>
    <build>
        <extensions>
            <extension>
                <groupId>kr.motd.maven</groupId>
                <artifactId>os-maven-plugin</artifactId>
                <version>1.4.1.Final</version>
            </extension>
        </extensions>
        <plugins>
            <plugin>
                <groupId>org.xolstice.maven.plugins</groupId>
                <artifactId>protobuf-maven-plugin</artifactId>
                <version>0.5.0</version>
                <configuration>
                    <protocArtifact>com.google.protobuf:protoc:3.0.0:exe:${os.detected.classifier}</protocArtifact>
                    <pluginId>grpc-java</pluginId>
                    <pluginArtifact>io.grpc:protoc-gen-grpc-java:1.0.0:exe:${os.detected.classifier}</pluginArtifact>
                </configuration>
                <executions>
                    <execution>
                        <goals>
                            <goal>compile</goal>
                            <goal>compile-custom</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
  • proto文件

    syntax = "proto3";
    option java_package = "com.qidai.proto";
    option java_outer_classname = "MyThing";
    
    message Request {
        string name = 1;
    }
    message Response {
        string name = 2;
    }
    service MyRPC {
        rpc sayHi(Request) returns(Response);
    }
  • 到这后maven插件compile,进行编译proto文件
  • Server

    import com.qidai.proto.MyRPCGrpc;
    import com.qidai.proto.MyThing;
    import io.grpc.ServerBuilder;
    import io.grpc.stub.StreamObserver;
    import java.io.IOException;
    public class Server {
        private static final int PORT = 2222;
        private final io.grpc.Server server;
        public Server() throws IOException {
            //这个部分启动server
            this.server = ServerBuilder.forPort(PORT)
                    .addService(new MyRPCImpl())
                    .build()
                    .start();
            System.out.println("Server Started ...");
        }
        private void stop() {
            if (server != null) {
                server.shutdown();
            }
        }
        private void blockUntilShutdown() throws InterruptedException {
            if (server != null) {
                server.awaitTermination();
            }
        }
        public static void main(String[] args) throws IOException, InterruptedException {
            Server server = new Server();
            //block Server防止关闭
            server.blockUntilShutdown();
        }
        static class MyRPCImpl extends MyRPCGrpc.MyRPCImplBase{
            @Override  //这个方法是在proto中定义的rpc方法名,client传来的rpc请求以怎样的逻辑处理
            public void sayHi(MyThing.Request request, StreamObserver<MyThing.Response> responseObserver) {
                System.out.println("request -> " + request.getName());
                MyThing.Response response = MyThing.Response.newBuilder()
                        .setName("Client Hello "+ request.getName())
                        .build();
                //包装返回信息
                responseObserver.onNext(response);
                //等待结束一次请求
                responseObserver.onCompleted();
            }
        }
    }
  • Client

    import com.qidai.proto.MyRPCGrpc;
    import com.qidai.proto.MyThing;
    import io.grpc.ManagedChannel;
    import io.grpc.ManagedChannelBuilder;
    import java.util.concurrent.TimeUnit;
    public class Client {
        private final ManagedChannelBuilder<?> managedChannelBuilder;
        private final MyRPCGrpc.MyRPCBlockingStub blockingStub;
        private final ManagedChannel channel;
        public Client(String name, int port) {
            managedChannelBuilder = ManagedChannelBuilder.forAddress(name, port);
            channel = managedChannelBuilder.usePlaintext().build();
            blockingStub = MyRPCGrpc.newBlockingStub(channel);
        }
        public void shutdown() throws InterruptedException {
            channel.shutdown().awaitTermination(5, TimeUnit.SECONDS);
        }
        public  void sayHi(String name){
            MyThing.Request request = MyThing.Request.newBuilder().setName(name).build();
            MyThing.Response response = blockingStub.sayHi(request);
            System.out.println(response.getName());
        }
        public static void main(String[] args) throws Exception{
            Client client = new Client("localhost", 2222);
            for (int i = 0; i < 10; i++) {
                Thread.sleep(1000);
                //进行rpc调用的真正逻辑
                client.sayHi("Hello Server -> " + i);
            }
            client.shutdown();
        }
    }
  • 到这了就可以使用了,先启动server然后启动client,效果如下

    Server
      request -> Hello Server -> 0
      request -> Hello Server -> 1
      request -> Hello Server -> 2
      request -> Hello Server -> 3
      request -> Hello Server -> 4
      request -> Hello Server -> 5
      request -> Hello Server -> 6
      request -> Hello Server -> 7
      request -> Hello Server -> 8
      request -> Hello Server -> 9  
    Client
      Client Hello Hello Server -> 0
      Client Hello Hello Server -> 1
      Client Hello Hello Server -> 2
      Client Hello Hello Server -> 3
      Client Hello Hello Server -> 4
      Client Hello Hello Server -> 5
      Client Hello Hello Server -> 6
      Client Hello Hello Server -> 7
      Client Hello Hello Server -> 8
      Client Hello Hello Server -> 9
目录
相关文章
|
编译器 Go
Golang 语言 gRPC 使用的接口设计语言 protobuf
Golang 语言 gRPC 使用的接口设计语言 protobuf
89 0
|
JSON 自然语言处理 编译器
Golang 语言 gRPC 到底是什么?
Golang 语言 gRPC 到底是什么?
95 0
|
6月前
|
XML JSON Java
GRPC与 ProtoBuf 的理解与总结
GRPC与 ProtoBuf 的理解与总结
234 0
|
自然语言处理 编译器 Go
Golang 语言 gRPC 怎么使用?
Golang 语言 gRPC 怎么使用?
70 0
|
8月前
|
JSON 缓存 Java
ProtoBuf-gRPC实践
gRPC 是一个高性能、开源、通用的 RPC 框架,基于 HTTP/2 设计,使用 Protocol Buffers(ProtoBuf)作为序列化协议。ProtoBuf 允许定义服务接口和消息类型,编译器会生成客户端和服务器端的代码,简化了跨平台的通信。 **gRPC 学习背景** 1. **为什么要学gRPC**:提高网络通信效率,支持多平台、多语言,提供高效序列化和反序列化,以及可靠的安全性和流控。 2. **RPC是什么**:远程过程调用,允许一个程序调用另一个不在同一设备上的程序。 3. **网络库收益分析**:gRPC 提供高效、安全、易用的网
411 0
|
8月前
|
Go
golang安装protoc和gRPC步骤
golang安装protoc和gRPC步骤
170 0
|
8月前
|
C++
vs2017下gRPC的编译以及简单使用
vs2017下gRPC的编译以及简单使用
164 0
|
JSON Linux 测试技术
go语言处理数据、基本通信以及环境配置 -- json,protobuf,grpc
go语言处理数据、基本通信以及环境配置 -- json,protobuf,grpc
|
测试技术 Go
gRPC源码分析(四):剖析Proto序列化
首先,针对读源码是先看源代码还是测试代码,因人而异。个人建议在对源码毫无头绪时,先从测试入手,了解大致功能;如果有一定基础,那么也可以直接入手源代码。我认为优秀的Go源码可读性是非常高的,所以一般情况下,我都直接从源文件入手,遇到问题才会去对应的测试里阅读。
135 1
|
负载均衡
gRPC源码分析(二):从官网文档看gRPC的特性
在第一部分,我们学习了gRPC的基本调用过程,这样我们对全局层面有了一定了解。接下来,我们将结合官方文档,继续深入学习、探索下去。
90 0