如何在Spring Boot微服务中集成gRPC通讯

一、首先声明gRPC接口

这里引入的是最新的gRpc-core 1.37版本, 采用的grcp-spring-boot-starter封装的版本进行实现,github地址:

https://github.com/yidongnan/grpc-spring-boot-starter

  1. 要实现gRpc通讯, 先定义接口以及入参出参信息

    syntax = "proto3";
    
    option java_multiple_files = true;
    option java_package = "com.grpc.spring.api.order";
    option java_outer_classname = "OrderServiceProto";
    
    // The service definition.
    service OrderService {
        // The service method
        rpc GetOrderInfo (GetOrderRequest) returns (GetOrderReply) {
        }
    }
    
    // The request message
    message GetOrderRequest {
        string orderNo = 1;
    }
    
    // The response message
    message GetOrderReply {
        string orderNo = 1;
        string userName = 2;
        string address = 3;
        int64 price = 4;
    }
    
  2. 定义好之后, 就需要通过protoc工具,生成对应的JAVA代码。

    为便于使用, 这里封装了bat脚本,并提供了对应的protoc执行程序。

    @echo off
    for %%i in (proto/*.proto) do (
      d:/TestCode/protoc.exe --plugin=protoc-gen-grpc-java=d:/TestCode/protoc-grpc.exe --java_out=../java --grpc-java_out=../java ./proto/%%i
      echo generate %%i to java file successfully!
    )
    

    该脚本意思, 会扫描当前proto目录下的所有proto脚本文件, 通过protoc-grpc插件, 在指定目录下生成gRpc通讯接口的JAVA代码。

  3. 生成后的JAVA代码
    file
    执行成功后, 会生成GPRC通讯接口, 入参和出参信息。

二、如何在服务端集成gRPC

  1. 首先整理配置好POM依赖,这里将共用的依赖抽取到父级POM文件中。

    父级POM依赖:

    <dependencies>
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>3.1.0</version>
        </dependency>
        <!-- spring boot grpc 依赖 -->
        <dependency>
            <groupId>net.devh</groupId>
            <artifactId>grpc-spring-boot-starter</artifactId>
            <version>${gprc.spring.version}</version>
        </dependency>
        <!-- jackson -->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>${jackson.verson}</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>${jackson.verson}</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-annotations</artifactId>
            <version>${jackson.verson}</version>
        </dependency>
        <!-- lombok 插件, 简化开发代码 -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
            <version>1.18.8</version>
        </dependency>
    </dependencies>
    
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>com.grpc.spring</groupId>
                <artifactId>grpc-spring-api</artifactId>
                <version>${grpc.api.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-parent</artifactId>
                <version>${spring.boot.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-web</artifactId>
                <version>${spring.boot.version}</version>
            </dependency>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>2020.0.2</version>
            </dependency>
            <dependency>
                <groupId>com.alibaba.cloud</groupId>
                <artifactId>spring-cloud-alibaba-dependencies</artifactId>
                <version>2021.1</version>
            </dependency>
        </dependencies>
    </dependencyManagement>
    
  2. 服务端POM配置

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>net.devh</groupId>
            <artifactId>grpc-server-spring-boot-starter</artifactId>
            <version>${gprc.spring.version}</version>
        </dependency>
        <dependency>
            <groupId>com.grpc.spring</groupId>
            <artifactId>grpc-spring-api</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
            <version>3.0.2</version>
            <exclusions>
                <exclusion>
                    <artifactId>servlet-api</artifactId>
                    <groupId>javax.servlet</groupId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>    
    

    作为服务端, 需要加入gRpc服务端的依赖grpc-server-spring-boot-starter, 同时引入eureka, 可以实现微服务之间的调用以及负载。

  3. 服务端接口实现

    这里为作测试验证, 定义个订单服务接口, GrpcOrderService:

    /**
     * 订单服务接口实现
     */
    @GrpcService
    public class GrpcOrderService extends OrderServiceGrpc.OrderServiceImplBase {
    
        @Value("${server.port}")
        private String serverPort;
    
        @Override
        public void getOrderInfo(GetOrderRequest request, StreamObserver<GetOrderReply> responseObserver) {
            GetOrderReply reply =GetOrderReply.newBuilder().setOrderNo(request.getOrderNo())
                    .setAddress("广东省深圳市,端口号" + serverPort)
                    .setUserName("tester")
                    .setPrice(System.currentTimeMillis())
                    .build();
            responseObserver.onNext(reply);
            responseObserver.onCompleted();
        }
    }
    

    要实现gRpc服务端接口, 必须做声明, 采用@GrpcService注解, 然后继承gRpc生成的接口OrderServiceGrpc,重新getOrderInfo获取订单信息接口, 实现具体的处理逻辑。

  4. 服务端的配置

    application.yml

    server:
      port: 18082
    spring:
      application:
        name: grpc-spring-server
    grpc:
      server:
        port: 0
    eureka:
      instance:
        prefer-ip-address: true
      client:
        register-with-eureka: true
        fetch-registry: false
        service-url:
          defaultZone: http://${eureka.instance.hostname:eureka.kyeapi.com}:${eureka.server.port:18761}/eureka/
    
    

    服务端的gRpc端口配置为0代表随机分配, gRpc服务信息会注册至eureka。

三、 如何在客户单集成gRPC:

  1. 客户端的POM依赖配置

    <dependencies>
        <dependency>
            <groupId>com.grpc.spring</groupId>
            <artifactId>grpc-spring-api</artifactId>
        </dependency>
        <dependency>
            <groupId>net.devh</groupId>
            <artifactId>grpc-client-spring-boot-starter</artifactId>
            <version>${gprc.spring.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
            <version>3.0.2</version>
        </dependency>
    
        <!-- 添加 Spring Boot 依赖 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
    
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
            <version>3.0.2</version>
            <exclusions>
                <exclusion>
                    <artifactId>servlet-api</artifactId>
                    <groupId>javax.servlet</groupId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>
    

    加入提供的客户端依赖grpc-client-spring-boot-starter,这里注意,添加的eureka-client依赖, 要排除servlet-api, 以免启动冲突。

  2. 客户端调用实现

    GrpcOrderClientService:

    /**
     *  订单信息获取客户端接口
     */
    @Service
    @Log4j2
    public class GrpcOrderClientService {
    
        @GrpcClient("grpc-server")
        private OrderServiceGrpc.OrderServiceBlockingStub orderServiceBlockingStub;
    
        /**
         * 获取订单信息
         * @param orderNo
         * @return
         */
        public String getOrderInfo(String orderNo) {
            try {
                //将远程调用的Grpc服务端接口信息, 返回给客户端。
                GetOrderRequest request  = GetOrderRequest.newBuilder().setOrderNo(orderNo).build();
                final GetOrderReply response = orderServiceBlockingStub.getOrderInfo(request);
                String result = "orderNo: " + response.getOrderNo() + ", userName: " + response.getUserName() + ", address: " + response.getAddress();
                return result;
            }catch (Exception e) {
                log.error(e.getMessage(), e);
                return "error! " + e.getMessage();
            }
        }
    }
    
    

    这里通过@GrpcClient注解, 引入gRpc接口,这里的gRpc服务端名称grpc-server要与配置文件的名称保持一致。

  3. 客户端配置

    application.yml

    server:
      port: 18080
    spring:
      application:
        name: grpc-spring-client
    grpc:
      client:
        grpc-server:
          address: 'discovery:///grpc-spring-server'
          enableKeepAlive: true
          keepAliveWithoutCalls: true
          negotiationType: plaintext
    eureka:
      instance:
        prefer-ip-address: true
        status-page-url-path: /actuator/info
        health-check-url-path: /actuator/health
      client:
        register-with-eureka: false
        fetch-registry: true
        service-url:
          defaultZone: http://${eureka.instance.hostname:eureka.kyeapi.com}:${eureka.server.port:18761}/eureka/
    
    

    这里定义了一个gRpc的连接信息grpc-server,与前面注解的名称保持一致, adress要配置gRpc服务端的名称, 不需要配置具体的IP和端口,会通过eureka来拉取服务端的具体连接信息。

四、 测试验证

  1. 启动服务

    启动Eureka服务、gRpc服务端与客户端,为便于测试负载, 这里启动两个服务端实例, 端口号分别为18081与108082。

  2. 调用验证

    第一调用:

    file

    第二次调用:

    file

    可以看到, 能够正常调用gRpc服务接口, 同时gRpc也是可以支持负载均衡。

最后, 需要的完整的源码, 可以从以下链接获取:

https://download.csdn.net/download/hxx688/22041124

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Spring Cloud 是一个基于 Spring Boot 的云计算解决方案,它提供了一系列工具和服务来简化分布式系统和微服务的开发、部署和管理。对于构建一个电商微服务架构,Spring Cloud 可以帮助你实现以下几个关键功能: 1. **服务发现**(Service Discovery):Spring Cloud Netflix Eureka 或 Consul 可以帮助你自动注册和发现服务实例,使得各个服务组件能够找到彼此。 2. **负载均衡**(Load Balancing):Hystrix 和 Ribbon 提供了流量管理和智能路由的功能,用于分摊请求到多个实例上,提高系统的可用性和性能。 3. **断路器模式**(Circuit Breaker):Hystrix 可以防止雪崩效应,即当某个服务不可用时,断开对该服务的请求,避免进一步的连锁失败。 4. **配置心**(Configuration Center):Spring Cloud Config 或 Zuul 可以集管理应用的配置,减少部署时的复杂性。 5. **API 网关**(API Gateway):Zuul 作为一个前端服务,可以处理 API 路由、安全控制和跨域策略等。 6. **服务间通信**(Service Communication):使用 RESTful 调用或 gRPC 进行服务间的通信,Spring Cloud Gateway 可以支持这些协议。 7. **消息队列**(Message Queue):RabbitMQ 或 Apache Kafka 可以作为异步通信的桥梁,提高系统的可扩展性和容错能力。 8. **服务编排**(Service Orchestration):Spring Cloud Stream 可以方便地集成消息间件,实现消息驱动的业务流程。 要构建一个电商微服务架构,你需要根据实际需求设计服务拆分,定义好服务接口和依赖,然后配置好 Spring Cloud 的各个组件。同时,可能还需要考虑数据一致性、事务管理、日志追踪、安全性等问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

麦神-mirson

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值