教你在Spring Boot微服务中集成gRPC通讯的方法

网友投稿 849 2022-09-28


教你在Spring Boot微服务中集成gRPC通讯的方法

一、首先声明gRPC接口

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

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

要实现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;

}

定义好之后, 就需要通过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代码。

生成后的JAVA代码

执行成功后, 会生成GPRC通讯接口, 入参和出参信息。

二、如何在服务端集成gRPC

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

父级POM依赖:

javax.servlet

javax.servlet-api

3.1.0

net.devh

grpc-spring-boot-starter

${gprc.spring.version}

com.fasterxml.jackson.core

jackson-databind

${jackson.verson}

com.fasterxml.jackson.core

jackson-core

${jackson.verson}

com.fasterxml.jackson.core

jackson-annotations

${jackson.verson}

org.projectlombok

lombok

true

1.18.8

com.grpc.spring

grpc-spring-api

${grpc.api.version}

org.springframework.boot

spring-boot-starter-parent

${spring.boot.version}

org.springframework.boot

spring-boot-starter-web

${spring.boot.version}

org.springframework.cloud

spring-cloud-dependencies

2020.0.2

com.alibaba.cloud

spring-cloud-alibaba-dependencies

2021.1

服务端POM配置

org.springframework.boot

spring-boot-starter-web

net.devh

grpc-server-spring-boot-starter

${gprc.spring.version}

com.grpc.spring

grpc-spring-api

org.springframework.cloud

spring-cloud-starter-netflix-eureka-client

3.0.2

servlet-api

javax.servlet

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

服务端接口实现

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

/**

* 订单服务接口实现

*/

@GrpcService

public class GrpcOrderService extends OrderServiceGrpc.OrderServiceImplBase {

@Value("${server.port}")

private String serverPort;

@Override

public void getOrderInfo(GetOrderRequest request, StreamObserver 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获取订单信息接口, 实现具体的处理逻辑。

服务端的配置

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。

ASHpcOiEB

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

客户端的POM依赖配置

com.grpc.spring

grpc-spring-api

net.devh

grpc-client-spring-boot-starter

${gprc.spring.version}

org.springframework.cloud

spring-cloud-starter-netflix-eureka-client

3.0.2

org.springframework.boot

spring-boot-starter-web

org.springframework.cloud

spring-cloud-starter-netflix-eureka-client

3.0.2

servlet-api

javax.servlet

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

客户端调用实现

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要与配置文件的名称保持一致。

客户端配置

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来拉取服务端的具体连接信息。

四、 测试验证

启动服务

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

调用验证

第一调用:

第二次调用:

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

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

http://xiazai.jb51.net/202109/yuanma/grpcv_jb51.rar


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:使用MD5加密后的字符串存密码安全吗?你不得不了解的Hash算法!(简单介绍MD5加密算法)
下一篇:BCS演讲实录 | 未来智安CTO陈毓端精讲《XDR扩展威胁检测响应探索与实践》
相关文章

 发表评论

暂时没有评论,来抢沙发吧~