当前位置:   article > 正文

gRPC在Spring Cloud中的应用_@grpcclient

@grpcclient

1、前言

    在微服务开发中,服务间的调用一般有两种方式:Feign、RestTemplate,但在实际使用过程中,尤其是Feign,存在各种限制及局限性,如:HTTP请求方式、返回类型等限制,有时会让你觉得那那都别扭。在微服务项目中,服务间的调用,是非常普遍频繁的,其性能也不是很理想。

   为了解决上述问题,进行反复对比,最终服务间的调用方式采取了gRPC方式,其显著特点就是性能之高(通信采用Netty),通过proto文件定义的接口也是非常清晰而又灵活。本文主要就gRPC在Spring Cloud项目中的使用进行说明实战。

  关于gRPC相关基础知识可以参考上一篇文章gRPC的使用

2、gRPC在Spring Cloud中的使用

    看过上一篇文章gRPC的使用的话,你就清楚如果直接使用gRPC,显得有些吃力,因此借助一些开源的框架变得尤为必要。gRPC在Spring Cloud中使用开源项目grpc-spring-boot-starter,便于在Spring Cloud项目中开发应用。

(grpc-spring-boot-starter虽然存在一些问题,但集成Sping Cloud项目已经相当高了,还是不错之选。如果你有时间,精力,还是又必要在源码基础上进行开发。)

下面以实际demo来说明grpc-spring-boot-starter的应用。

2.1 特点

  • 使用@ GrpcService自动创建并运行一个 gRPC 服务,内嵌在 spring-boot 应用中
  • 使用@ GrpcClient自动创建和管理你的客户端
  • 支持Spring Cloud(向Consul或Eureka注册服务并获取gRPC服务器信息)
  • 支持Spring Sleuth 进行链路跟踪
  • 支持对于server、client 分别设置全局拦截器或单个的拦截器
  • 支持Spring-Security
  • 支持metric (micrometer / actuator)

(看了上面这些特点,就知道为啥选择这个开源项目了)

2.2 使用DEMO

2.2.1 定义gRPC接口

     基于protobuf来声明数据模型和RPC接口服务。

     创建一个公共字模块项目spring-boot-grpc-common,用于定义存放gRPC接口(proto),便于gRPC服务端和客户端使用。以helloworld.proto(src\main\proto\helloworld.proto)为例:

  1. syntax = "proto3";
  2. option java_multiple_files = true;
  3. option java_package = "com.xcbeyond.springboot.grpc.lib";
  4. option java_outer_classname = "HelloWorldProto";
  5. // The greeting service definition.
  6. service Simple {
  7. // Sends a greeting
  8. rpc SayHello (HelloRequest) returns (HelloReply) {
  9. }
  10. }
  11. // The request message containing the user's name.
  12. message HelloRequest {
  13. string name = 1;
  14. }
  15. // The response message containing the greetings
  16. message HelloReply {
  17. string message = 1;
  18. }

     根据proto的命令可以转换成对应的语言的代码,生成java代码,也可以借助maven插件,在编译时自动生成。这里通过mavent插件,可以在pom.xml中增加如下依赖:

  1. <build>
  2. <extensions>
  3. <extension>
  4. <groupId>kr.motd.maven</groupId>
  5. <artifactId>os-maven-plugin</artifactId>
  6. <version>${os.plugin.version}</version>
  7. </extension>
  8. </extensions>
  9. <plugins>
  10. <plugin>
  11. <groupId>org.xolstice.maven.plugins</groupId>
  12. <artifactId>protobuf-maven-plugin</artifactId>
  13. <version>${protobuf.plugin.version}</version>
  14. <configuration>
  15. <protocArtifact>com.google.protobuf:protoc:${protoc.version}:exe:${os.detected.classifier}</protocArtifact>
  16. <pluginId>grpc-java</pluginId>
  17. <pluginArtifact>io.grpc:protoc-gen-grpc-java:${grpc.version}:exe:${os.detected.classifier}</pluginArtifact>
  18. </configuration>
  19. <executions>
  20. <execution>
  21. <goals>
  22. <goal>compile</goal>
  23. <goal>compile-custom</goal>
  24. </goals>
  25. </execution>
  26. </executions>
  27. </plugin>
  28. </plugins>
  29. </build>

在maven中进行package编译打包,将会在target中看见根据proto自动生成的Java类。(编译过程中可能会报错,此时可以忽略)

2.2.2 gRPC服务端

maven依赖:

  1. <dependency>
  2. <groupId>net.devh</groupId>
  3. <artifactId>grpc-server-spring-boot-starter</artifactId>
  4. <version>2.2.1.RELEASE</version>
  5. </dependency>

实现 gRPC server 的业务逻辑,使用注解@GrpcService定义gRPC服务端,如下所示:

  1. package com.xcbeyond.springboot.grpc.server.service;
  2. import com.xcbeyond.springboot.grpc.lib.HelloReply;
  3. import com.xcbeyond.springboot.grpc.lib.HelloRequest;
  4. import com.xcbeyond.springboot.grpc.lib.SimpleGrpc;
  5. import io.grpc.stub.StreamObserver;
  6. import net.devh.boot.grpc.server.service.GrpcService;
  7. /**
  8. * @Auther: xcbeyond
  9. * @Date: 2019/3/6 18:15
  10. */
  11. @GrpcService
  12. public class GrpcServerService extends SimpleGrpc.SimpleImplBase {
  13. @Override
  14. public void sayHello(HelloRequest request, StreamObserver<HelloReply> responseObserver) {
  15. System.out.println("GrpcServerService...");
  16. HelloReply reply = HelloReply.newBuilder().setMessage("Hello ==> " + request.getName()).build();
  17. responseObserver.onNext(reply);
  18. responseObserver.onCompleted();
  19. }
  20. }

application.yml配置:

gRPC 的 host 跟 port ,默认的监听的 host 是 0.0.0.0,默认的 port 是 9090,配置为0将会自动分配未使用的端口。

  1. grpc:
  2. server:
  3. port: 0

 

2.2.3 gRPC客户端

maven依赖:

  1. <dependency>
  2. <groupId>net.devh</groupId>
  3. <artifactId>grpc-client-spring-boot-starter</artifactId>
  4. <version>2.2.1.RELEASE</version>
  5. </dependency>

使用注解@GrpcClient来调用服务端接口,通过

HelloReply response = simpleBlockingStub.sayHello(HelloRequest.newBuilder().setName(name).build());

直接向服务端发起请求,和调用本地接口一样。

  1. package com.xcbeyond.springboot.grpc.client.service;
  2. import com.xcbeyond.springboot.grpc.lib.HelloReply;
  3. import com.xcbeyond.springboot.grpc.lib.HelloRequest;
  4. import com.xcbeyond.springboot.grpc.lib.SimpleGrpc.SimpleBlockingStub;
  5. import io.grpc.StatusRuntimeException;
  6. import net.devh.boot.grpc.client.inject.GrpcClient;
  7. import org.springframework.stereotype.Service;
  8. /**
  9. * @Auther: xcbeyond
  10. * @Date: 2019/3/7 09:10
  11. */
  12. @Service
  13. public class GrpcClientService {
  14. @GrpcClient("spring-boot-grpc-server")
  15. private SimpleBlockingStub simpleBlockingStub;
  16. public String sendMessage(String name) {
  17. try {
  18. HelloReply response = simpleBlockingStub.sayHello(HelloRequest.newBuilder().setName(name).build());
  19. return response.getMessage();
  20. } catch (final StatusRuntimeException e) {
  21. return "FAILED with " + e.getStatus().getCode();
  22. }
  23. }
  24. }

application.yml配置:

   spring-boot-grpc-server,即:服务端应用名,结合spring cloud Eureka注册中心,通过服务名将会找到服务端的ip,进行通信,实际上是netty通信。
  1. grpc:
  2. client:
  3. spring-boot-grpc-server:
  4. enableKeepAlive: true
  5. keepAliveWithoutCalls: true
  6. negotiationType: plaintext

本demo完整代码请参考https://github.com/xcbeyond/spring-boot-grpc.git

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop】
推荐阅读
相关标签
  

闽ICP备14008679号