gRPC入门-Hello World

本文是gRPC的一个简单例子,以protocol buffers 3作为契约类型,使用gRPC自动生成服务端和客户端代码,实现服务的远程调用。

示例代码-Github地址

gRPC
  • 创建gradle类型工程,build.gradle文件如下,包含了根据.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 {}
            }
        }
    }
    
  • 创建目录src/main/proto,并在其中新建契约文件helloworld.proto。这里定义一个请求类型HelloRequest、一个响应类型HelloReply,和一个简单的服务(serviceGreeterGreeter只提供一个简单的RPC服务(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;
    }
    
  • 运行命令gradle clean build,在工程目录下会生成 build文件夹,其中generated目录下包含由插件io.grpc:protoc-gen-grpc-java所生成的RPC代码。

    生成的代码目录

  • 生成的GreeterGrpc.java中包含自定义服务需要继承的抽象类GreeterImplBase,以及stub
    GreeterStub(异步返回response)和GreeterBlockingStub(同步等待response)。客户端通过stub调用服务端。

    image.png

  • 定义自己的服务,继承GreeterImplBase并重写契约中定义的服务sayHello,实现真正的业务逻辑。onNext方法用来返回 helloReply对象给客户端,onCompleted方法用来标明此次RPC调用已结束。

    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();
        }
    }
    
  • 创建好自定义服务后,就可以新建和启动一个服务器,用来接收客户端的连接。
    1. 使用ServerBuilder创建服务器,forPort方法监听端口
    2. 创建MyService的一个实例,并传递给ServerBuilderaddService方法
    3. 调用buildstart启动服务器

    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();
            }
        }
    }
    
  • 创建客户端:

    1.创建gRPC channel,将stub与服务器连接:这里使用ManagedChannelBuilder来指定主机和端口

    1. 将创建的channel作为参数,创建stub

    2. 使用stub像调用本地方法一样调用远程服务

      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仓库后,运行gradle clean build生成build文件夹,并将main目录设置为source root

main目录设置为Sources

2. .proto文件默认需要放在src/main/proto下面,否则无法生成代码。另外可以自定义存放目录:
自定义.proto文件的存放目录

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容

  • gRPC 是一个高性能、通用的开源RPC框架,基于HTTP/2协议标准和Protobuf序列化协议开发,支持众多的...
    小波同学阅读 19,392评论 6 19
  • GRPC是基于protocol buffers3.0协议的. 本文将向您介绍gRPC和protocol buffe...
    二月_春风阅读 17,972评论 2 28
  • 1.简介 在gRPC中,客户端应用程序可以直接调用不同计算机上的服务器应用程序上的方法,就像它是本地对象一样,使您...
    第八共同体阅读 1,852评论 0 6
  • rpc框架原理 RPC 框架的目标就是让远程服务调用更加简单、透明,RPC 框架负责屏蔽底层的传输方式(TCP 或...
    tracy_668阅读 9,950评论 0 7
  • 什么是 RPC 这个是知乎上的一个关于 rpc 的问题谁能用通俗的语言解释一下什么是 RPC 框架? gRPC的官...
    简简单单敲代码阅读 1,864评论 0 1