gRPC入門-Hello World

本文是gRPC的一個簡單例子,以protocol buffers 3作為契約類型,使用gRPC自動生成服務(wù)端和客戶端代碼,實現(xiàn)服務(wù)的遠(yuǎn)程調(diào)用。

示例代碼-Github地址

gRPC
  • 創(chuàng)建gradle類型工程,build.gradle文件如下,包含了根據(jù).proto自動生成代碼的插件.

    apply plugin: 'java'
    apply plugin: 'com.google.protobuf'
    
    buildscript {
        repositories {
            maven {
                url "http://maven.aliyun.com/nexus/content/groups/public/" }
        }
        dependencies {
            classpath 'com.google.protobuf:protobuf-gradle-plugin:0.8.5'
        }
    }
    
    repositories {
        maven {
            url "http://maven.aliyun.com/nexus/content/groups/public/" }
        mavenLocal()
    }
    
    def grpcVersion = '1.14.0'
    def nettyTcNativeVersion = '2.0.7.Final'
    def protobufVersion = '3.5.1'
    
    dependencies {
        compile "com.google.api.grpc:proto-google-common-protos:1.0.0"
        compile "io.grpc:grpc-alts:${grpcVersion}"
        compile "io.grpc:grpc-netty:${grpcVersion}"
        compile "io.grpc:grpc-protobuf:${grpcVersion}"
        compile "io.grpc:grpc-stub:${grpcVersion}"
        compileOnly "javax.annotation:javax.annotation-api:1.2"
        compile "io.netty:netty-tcnative-boringssl-static:${nettyTcNativeVersion}"
        compile "com.google.protobuf:protobuf-java-util:${protobufVersion}"
    }
    
    protobuf {
        protoc {
            artifact = "com.google.protobuf:protoc:3.5.1-1"
        }
        plugins {
            grpc {
                artifact = 'io.grpc:protoc-gen-grpc-java:1.15.0'
            }
        }
        generateProtoTasks {
            all()*.plugins {
                grpc {}
            }
        }
    }
    
  • 創(chuàng)建目錄src/main/proto,并在其中新建契約文件helloworld.proto。這里定義一個請求類型HelloRequest、一個響應(yīng)類型HelloReply,和一個簡單的服務(wù)(serviceGreeter。Greeter只提供一個簡單的RPC服務(wù)(simple RPCsayHello

    syntax = "proto3";
    
    package com.mattie.grpc;
    
    option java_package = "com.mattie.grpc";
    option java_outer_classname = "HelloWorldProtos";
    
    service Greeter {
      rpc SayHello (HelloRequest) returns (HelloReply) {}
    }
    
    message HelloRequest {
      string message = 1;
    }
    
    message HelloReply {
      string message = 1;
    }
    
  • 運(yùn)行命令gradle clean build,在工程目錄下會生成 build文件夾,其中generated目錄下包含由插件io.grpc:protoc-gen-grpc-java所生成的RPC代碼。

    生成的代碼目錄

  • 生成的GreeterGrpc.java中包含自定義服務(wù)需要繼承的抽象類GreeterImplBase,以及stub。
    GreeterStub(異步返回response)和GreeterBlockingStub(同步等待response)??蛻舳送ㄟ^stub調(diào)用服務(wù)端。

    image.png

  • 定義自己的服務(wù),繼承GreeterImplBase并重寫契約中定義的服務(wù)sayHello,實現(xiàn)真正的業(yè)務(wù)邏輯。onNext方法用來返回 helloReply對象給客戶端,onCompleted方法用來標(biāo)明此次RPC調(diào)用已結(jié)束。

    public class MyService extends GreeterImplBase {
        @Override
        public void sayHello(HelloRequest request, StreamObserver<HelloReply> responseObserver) {
            HelloReply helloReply = HelloReply.newBuilder().setMessage("hello client.").build();
            responseObserver.onNext(helloReply);
            responseObserver.onCompleted();
        }
    }
    
  • 創(chuàng)建好自定義服務(wù)后,就可以新建和啟動一個服務(wù)器,用來接收客戶端的連接。
    1. 使用ServerBuilder創(chuàng)建服務(wù)器,forPort方法監(jiān)聽端口
    2. 創(chuàng)建MyService的一個實例,并傳遞給ServerBuilderaddService方法
    3. 調(diào)用buildstart啟動服務(wù)器

    public class MyServer {
    
        public static void main(String[] args) {
            ServerBuilder<?> serverBuilder = ServerBuilder.forPort(8899);
            serverBuilder.addService(new MyService());
            Server server = serverBuilder.build();
            try {
                server.start();
                server.awaitTermination();
            } catch (IOException | InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    
  • 創(chuàng)建客戶端:

    1.創(chuàng)建gRPC channel,將stub與服務(wù)器連接:這里使用ManagedChannelBuilder來指定主機(jī)和端口

    1. 將創(chuàng)建的channel作為參數(shù),創(chuàng)建stub

    2. 使用stub像調(diào)用本地方法一樣調(diào)用遠(yuǎn)程服務(wù)

      public class MyClient {

       public static void main(String[] args) {
           //使用usePlaintext,否則使用加密連接
           ManagedChannelBuilder<?> channelBuilder = ManagedChannelBuilder.forAddress("localhost", 8899).usePlaintext();
           ManagedChannel channel = channelBuilder.build();
      
           GreeterGrpc.GreeterBlockingStub blockingStub = GreeterGrpc.newBlockingStub(channel);
           HelloWorldProtos.HelloReply helloReply = blockingStub.sayHello(HelloWorldProtos.HelloRequest.newBuilder().setMessage("hello wolrd").build());
           System.out.println(helloReply.getMessage());
       }
      

      }

注:1. clone倉庫后,運(yùn)行gradle clean build生成build文件夾,并將main目錄設(shè)置為source root

main目錄設(shè)置為Sources

2. .proto文件默認(rèn)需要放在src/main/proto下面,否則無法生成代碼。另外可以自定義存放目錄:
自定義.proto文件的存放目錄

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • gRPC 是一個高性能、通用的開源RPC框架,基于HTTP/2協(xié)議標(biāo)準(zhǔn)和Protobuf序列化協(xié)議開發(fā),支持眾多的...
    小波同學(xué)閱讀 19,814評論 6 19
  • GRPC是基于protocol buffers3.0協(xié)議的. 本文將向您介紹gRPC和protocol buffe...
    二月_春風(fēng)閱讀 18,207評論 2 28
  • 1.簡介 在gRPC中,客戶端應(yīng)用程序可以直接調(diào)用不同計算機(jī)上的服務(wù)器應(yīng)用程序上的方法,就像它是本地對象一樣,使您...
    第八共同體閱讀 2,083評論 0 6
  • rpc框架原理 RPC 框架的目標(biāo)就是讓遠(yuǎn)程服務(wù)調(diào)用更加簡單、透明,RPC 框架負(fù)責(zé)屏蔽底層的傳輸方式(TCP 或...
    tracy_668閱讀 10,307評論 0 7
  • 什么是 RPC 這個是知乎上的一個關(guān)于 rpc 的問題誰能用通俗的語言解釋一下什么是 RPC 框架? gRPC的官...
    簡簡單單敲代碼閱讀 1,909評論 0 1

友情鏈接更多精彩內(nèi)容