0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

什么是gRPC

科技绿洲 来源:Java技术指北 作者:Java技术指北 2023-10-07 16:24 次阅读
  • 相信大家对RPC协议都有一定的了解,并且或多或少都会在项目中涉及,但可能都和小编类似,都是直接使用平台封装的插件,对于其中的原理不是很了解,今天借此机会和大家分享下最近接触的RPC框架-grpc,一同聊聊那些知其然却不知其所以然的内容。

概述

  • RPC(Remote Procedure Call)远程过程调用协议,是一种本地可以通过网络请求远程计算机,完成计算机间的数据内容的交互的协议,不需要了解网络底层技术就可以快速上手,使得开发更加容易,同时提升了交互体验效率。
  • 为了方便开发,有很多基于RPC协议实现的RPC框架,比如Thrift、Dubbo,和本文即将要介绍的gRPC。

什么是gRPC

  • gRPC是由google开发的一种支跨平台(语言)、高性能、开源通用的RPC框架。
  • 它是基于HTTP2.0协议的,可以保持客户端与服务端长连接,基于二进制流(字节流)传输数据。
  • 客户端与服务端交互过程
    • 客户端(gRPC Sub)调用A方法,发起RPC请求
    • 请求内容使用Protobf进行对象序列化压缩
    • 服务端(gRPC Server)接收请求,解析请求内容,业务处理后返回
    • 响应结果通过Protobuf进行对象序列化压缩
    • 客户端接收响应,解析响应内容,最终完成交互

实践案例

小编以java版进行案例展示,其它语言类似,可自行测试

  • POM依赖
    • gRPC官方提供完成的依赖配置,按照说明直接引用即可(依赖包含插件),版本仅供参考,也可选择其它版本。
< !-- gRPC配置 -- >
        < dependency >
            < groupId >io.grpc< /groupId >
            < artifactId >grpc-netty-shaded< /artifactId >
            < version >1.29.0< /version >
        < /dependency >
        < dependency >
            < groupId >io.grpc< /groupId >
            < artifactId >grpc-protobuf< /artifactId >
            < version >1.29.0< /version >
        < /dependency >
        < dependency >
            < groupId >io.grpc< /groupId >
            < artifactId >grpc-services< /artifactId >
            < version >1.29.0< /version >
        < /dependency >
        < dependency >
            < groupId >io.grpc< /groupId >
            < artifactId >grpc-stub< /artifactId >
            < version >1.29.0< /version >
        < /dependency >

        < !-- proto插件 -- >
                < plugins >
            < plugin >
                < groupId >org.xolstice.maven.plugins< /groupId >
                < artifactId >protobuf-maven-plugin< /artifactId >
                < version >0.6.1< /version >
                < configuration >
                    < protocArtifact >com.google.protobuf:protoc:3.11.0:exe:${os.detected.classifier}< /protocArtifact >
                    < pluginId >grpc-java< /pluginId >
                    < pluginArtifact >io.grpc:protoc-gen-grpc-java:1.29.0:exe:${os.detected.classifier}< /pluginArtifact >
                < /configuration >
                < executions >
                    < execution >
                        < goals >
                            < goal >compile< /goal >
                            < goal >compile-custom< /goal >
                        < /goals >
                    < /execution >
                < /executions >
            < /plugin >
        < /plugins >
  • 编写protobuf文件
    • 小编使用的是proto3版本,需要注意固定的目录结构(src/proto/*.proto),否则会编译失败。
    • proto文件有固定的编写格式,可以自行上网搜索即可
syntax = "proto3";
//包所在路径
option java_package = "com.greatom.dockerdemo.rule";
option java_multiple_files = true;
package rule;
//声明服务和方法
service RuleService {
    //查询并更新规则
    rpc getArchivesDic (RuleRequest) returns (RuleResponse);
    //获取当前规则字典
    rpc getRule (Request) returns (Response);
}
//定义请求对象
message RuleRequest {
    //    message RuleRPCDTO {
    //        int32 ruleCode = 1;
    //        string administrativeCost = 2;
    //    }
    Response ruleRPCDTO = 1;
    int32 basicId = 2;
}
//定义响应对象
message RuleResponse {
    int32 id = 1;
}
message Request {
}
//定义响应消息
message Response {
    int32 ruleCode = 1;
    string administrativeCost = 2;
}
  • 使用maven插件编译,双击执行(生成Bean,maven->Plugins->protobuf->protobuf:compile;生成具体接口,maven->Plugins->protobuf->protobuf:compile-custom)。
  • 小编只执行protobuf:compile命令,然后在target目录(targetgenerated-sourcesprotobuf)下就找到了生成的java文件,复制出来粘贴到项目执行目录下即可。
  • 编写接口实现类
  • 编译完后会生成RuleServiceGrpc接口,接下来就是按照自己的业务需求编写逻辑即可。小编定义的两个接口分别是 getArchivesDic(更新规则)、getRule(查询规则)。具体实现如下
// 继承生成的RuleServiceGrpc.RuleServiceImplBase
// 实现接口具体逻辑
@Component
public class RuleGRPCServer extends RuleServiceGrpc.RuleServiceImplBase {
    // 更新规则字典
    @Override
    public void getArchivesDic(RuleRequest request, StreamObserver< RuleResponse > responseObserver) {
        Response ruleRPCDTO = request.getRuleRPCDTO();
        RuleDTO ruleDTO = new RuleDTO();
        BeanUtils.copyProperties(ruleRPCDTO, ruleDTO);
        RuleResponse ruleResponse = RuleResponse.newBuilder().setId(1).build();
        responseObserver.onNext(ruleResponse);
        responseObserver.onCompleted();
    }
    // 查询规则字典
    @Override
    public void getRule(Request request, StreamObserver< Response > responseObserver) {
        Response response = Response.newBuilder().setRuleCode(1)
                .setAdministrativeCost("2222").build();
        responseObserver.onNext(response);
        responseObserver.onCompleted();
    }
}
  • 服务端与客户端
    • 服务端启动类
public static void main(String[] args) throws Exception {
        // 设置service接口.
        Server server = ServerBuilder.forPort(9999).addService(new RuleGRPCServiceImpl()).build().start();
        System.out.println(String.format("GRpc服务端启动成功, 端口号: %d.", port));
        server.awaitTermination();
    }

    日志 --- GRpc服务端启动成功, 端口号: 9999.
  • 客户端启动类
public static void main(String[] args) throws Exception {
        // 1. 拿到一个通信的channel
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9999).usePlaintext().build();
        try {
            // 2.拿到道理对象
            RuleServiceGrpc.RuleServiceBlockingStub rpcDateService = RuleServiceGrpc.newBlockingStub(managedChannel);
            Request rpcDateRequest = Request
                    .newBuilder()
                    .build();
            // 3. 请求
            Response rpcDateResponse = rpcDateService.getRule(rpcDateRequest);
            // 4. 输出结果
            System.out.println(rpcDateResponse.getRuleCode());
        } finally {
            // 5.关闭channel, 释放资源.
            managedChannel.shutdown();
        }
    }

    日志:
    - 16:05:44.628 [grpc-nio-worker-ELG-1-2] DEBUG io.grpc.netty.shaded.io.grpc.netty.NettyClientHandler - [id: 0x8447cc92, L:/127.0.0.1:60973 - R:localhost/127.0.0.1:9999] INBOUND DATA: streamId=3 padding=0 endStream=false length=12 bytes=0000000007086f1203323232
    - 16:05:44.648 [grpc-nio-worker-ELG-1-2] DEBUG io.grpc.netty.shaded.io.grpc.netty.NettyClientHandler - [id: 0x8447cc92, L:/127.0.0.1:60973 - R:localhost/127.0.0.1:9999] INBOUND HEADERS: streamId=3 headers=GrpcHttp2ResponseHeaders[grpc-status: 0] padding=0 endStream=true
    - 输出结果-----111
    - 16:05:44.664 [grpc-nio-worker-ELG-1-2] DEBUG io.grpc.netty.shaded.io.grpc.netty.NettyClientHandler - [id: 0x8447cc92, L:/127.0.0.1:60973 - R:localhost/127.0.0.1:9999] OUTBOUND GO_AWAY: lastStreamId=0 errorCode=0 length=0 bytes=
  • 客户端日志输出结果即表示客户端通过gRPC调用服务端成功,并返回结果。

总结

  • gRPC本质上就是传统的C|S模型,这样看角色分的清楚,也很容易理解。
  • 还有就是它很聪明的点是基于HTTP2.0协议的,而不是自己制定,这就对未来的网络开发很友好,降低了门槛。
  • 比较难上手的点在于proto文件的编写和使用,这部分需要插件等依赖,过程相对复杂,但也可能会出现工具或脚本,可以简化下这部分。但生成代码确实是真香~ 减少了一部分工作量。
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 计算机
    +关注

    关注

    19

    文章

    6649

    浏览量

    84533
  • 网络
    +关注

    关注

    14

    文章

    7251

    浏览量

    87441
  • 客户端
    +关注

    关注

    1

    文章

    282

    浏览量

    16339
  • 编译
    +关注

    关注

    0

    文章

    615

    浏览量

    32392
收藏 人收藏

    评论

    相关推荐

    GRPC的基础使用方法

    gRPC 是 Google 开源的高性能、通用的 RPC 框架,它采用了基于 HTTP/2 协议的二进制传输协议,支持多种语言,包括 Rust。Rust 语言 GRPC 模块是一个用于 Rust
    的头像 发表于 09-19 16:08 506次阅读

    153.RPC:11.3 gRPC说明

    服务器代码
    充八万
    发布于 :2023年07月12日 19:39:46

    REST、gRPC、GraphQL及WebHook的对比和选型

    首先REST--Resource Representational State Transfer, 中文直译就是资源在网络中以某种表现形式进行状态转移。
    发表于 06-13 10:34 2300次阅读

    Dubbo 在跨语言和协议穿透性方向的探索:支持 HTTP/2 gRPC

    Dubbo 在跨语言和协议穿透性方向上的探索:支持 HTTP/2 gRPC 和 Protobuf 本文整理自刘军在 Dubbo 成都 meetup 上分享的《Dubbo 在多语言和协议穿透性方向
    发表于 12-02 23:07 3680次阅读
    Dubbo 在跨语言和协议穿透性方向的探索:支持 HTTP/2 <b class='flag-5'>gRPC</b>

    谷歌开源高性能通用RPC框架gRPC

    谷歌开源了 gRPC-Kotlin/JVM,让开发者可以在 Kotlin 项目中更方便地使用 gRPC,以更简单的方式构建可靠的网络连接服务。
    的头像 发表于 04-20 14:43 2562次阅读
    谷歌开源高性能通用RPC框架<b class='flag-5'>gRPC</b>

    现代的服务端技术栈:Golang/Protobuf/gRPC详解

    Golang又称Go语言,是一个开源的、多用途的编程语言,由Google研发,并由于种种原因,正在日益流行。Golang已经有10年的历史,并且据Google称已经在生产环境中使用了接近7年的时间,这一点可能让大多数人大跌眼镜。
    的头像 发表于 12-25 17:32 806次阅读

    IP知识百科之什么是gRPC

    gRPC Google远程过程调用(Google Remote Procedure Call,gRPC)协议是谷歌发布的高性能、通用的开源RPC软件框架。gRPC提供了多种编程语言,同时gRP
    的头像 发表于 11-16 15:13 2814次阅读

    gWhisper gRPC命令行工具

    ./oschina_soft/gWhisper.zip
    发表于 06-15 09:28 1次下载
    gWhisper <b class='flag-5'>gRPC</b>命令行工具

    gRPC-Nebula微服务框架

    ./oschina_soft/grpc-nebula.zip
    发表于 06-22 14:59 0次下载
    <b class='flag-5'>gRPC</b>-Nebula微服务框架

    gRPC-Web访问gRPC服务的Web客户端

    ./oschina_soft/grpc-web.zip
    发表于 06-22 09:25 0次下载
    <b class='flag-5'>gRPC</b>-Web访问<b class='flag-5'>gRPC</b>服务的Web客户端

    正确使用gRPC与GraphQL

    TLDR:使用 GraphQL 进行客户端-服务器通信,使用 gRPC 进行服务器到服务器通信。有关此规则的例外情况,请参阅“判定”部分。 我已经阅读了很多关于这两种协议的比较,并想写一个全面和公正
    的头像 发表于 12-08 15:47 830次阅读

    搞一个grpc动态代理的想法,并初步实现

    本人之前很长一段时间从事saas,paas的开发。对于一些服务而言,既要提供grpc访问的能力,也要对外提供http访问的能力(做saas就是这么卑微)。并且这种需求通常不是一开始就提出来的
    的头像 发表于 02-08 16:13 1168次阅读

    gRPC内存马研究与查杀

    远程过程调用(Remote Procedure Call,缩写为 RPC)是一个计算机通信协议。
    的头像 发表于 06-01 11:24 1099次阅读
    <b class='flag-5'>gRPC</b>内存马研究与查杀

    使用go语言实现一个grpc拦截器

    在开发grpc服务时,我们经常会遇到一些通用的需求,比如:日志、链路追踪、鉴权等。这些需求可以通过grpc拦截器来实现。本文使用go语言来实现一个 grpc一元模式(Unary)拦截器,上报链路追踪信息。
    的头像 发表于 12-18 10:13 241次阅读
    使用go语言实现一个<b class='flag-5'>grpc</b>拦截器

    RPC的机制的诞生和基础概念

    客户端在调用远程方法时会使用阻塞式存根,所以gRPC主要使用同步的方式通信,在建立连接后,可以使用流的方式操作。
    的头像 发表于 03-11 10:06 155次阅读
    RPC的机制的诞生和基础概念