gRpc入门和springboot整合

news2025/1/15 23:35:35

gRpc入门和springboot整合

一、简介

1、gprc概念

gRpc是有google开源的一个高性能的pc框架,Stubby google内部的rpc,2015年正式开源,云原生时代一个RPC标准。

tips:异构系统,就是不同编程语言的系统。

2、grpc核心设计思路

grpc核心设计思路
 	1. 网络通信 --> grpc自己封装网络通信的部分,提供多种语言的网络通信的封装(java(netty),c,go)的支持异构语言。
 	2. 协议 --> http2协议,传输数据,支持二进制。支持双向流(双工)。连接的多路复用(NIO)
 	3. 序列化 --> 基于文本Json,基于二进制(java原生序列化方式,Thrift二进制序列化,压缩二进制序列化)
    			protobuf(protocol buffers)google开源的一种序列化方式。
    			dubbo也可以用protobuf。  时间效率和空间效率是Json的3---5倍。
    			定义了一套,IDL语言。
    4.代理的创建 --> 让调用者像调用本地方法那样,去调用远端的服务方法一样。
    				stub.  

3、gRpc和protobuf的区别

grpc是rpc框架。

protobuf是一种序列化方式

4、gRpc与ThrifRpc区别


共性:支持异构语言的	RPC
区别:
 	1. 网络通信  Trift TCP 专属协议   
 				grpc http2
 	2. 性能角度,thrift Rpc效率高于gRpc.
 	3. gRpc 大厂背书(google),云原生时代 合作更好 集成更多,所以grpc应用更广泛

5、gRpc的好处

  1. 高效的进行进程间通信。

  2. 支持多种语言 。原生支持,C GO Java实现。c语言版本上扩展,c++,c#,node JS,python,ruby,php (二等公民)

  3. 支持多平台运行。 Linux Andriod IOS MacOS Windows

  4. gRpc序列化方式采用protobuf,效率高

  5. 使用http2协议

  6. 大厂背书

二、http2.0协议

1、回顾 http1.x 协议

1.1、http1.0 协议

  • 请求响应的模式。

  • 短连接协议(无状态协议)。建立在tcp(长连接)上的,http主动断开连接(之前设备不好,主动断开)。

  • 传输数据文本结构

  • 单工,无法实现服务端推送。

    • 变相实现服务推送,(采用客户端轮询)

1.2、http1.1 协议

  • 请求相应的模式
  • 有限的长连接(保持一段时间)
  • 只能通过升级的方式,websocket协议,实现服务器想客户端推送。不属于http1.1的能力

1.3、总结http1.x协议特性:

  • 传输数据文本格式,可读性好但是效率差。
  • 本质上http1.x协议,无法实现双工通信。
  • 资源的请求。需要发送多次请求,建立多个连接才可以完成。分流(静态资源分离,cdn缓存加速)

2、HTTP2.0协议

  1. Http2.0协议是一个二进制协议,效率高于Http1.x协议,可读性差。
  2. 可以实现双工通信
  3. 一个连接可以请求多个数据。【多路复用】

3、HTTP2.O协议的三个概念

  1. 数据流 Stream
  2. 消息 message
  3. 帧 frame

一个连接,有多个数据流。Stream.

3.1、requestMessage

一个数据流,有request Message.

Message,里面有frame.

第一个frame,对应的就是header。

  • method:post
  • version:http2.0
  • user-agent:chrome

第二个frame,是body

  • data

如果get,就是一个frame帧。没有data,body。

3.1、responseMessage

第一个frame,对应的就是head。

  • status: 200
  • version:http2.0

第二个frame,是body

  • response数据

6b4d2c4df1a6c094f5b3d39a578981a

4、其他的相关概念

4.1、数据流的优先级

1. 数据流的优先级:可以为不同的Stream设置权重,来限制不同流的顺序。
1. 流控:client发送数据太快了,server处理不过来,通知client暂停数据的发送。

三、Protocol Buffers [protobuf]

1、简介

  1. protobuf是一种与编程语言无关【IDL】,与具体的平台无关【OS】。他定义的中间语言,可以方便,在Client和server之间进行RPC的数据传输。

  2. protobuf

    • protobuf 2
    • protobuf 3

    目前主流应用的都是protobuf3.

  3. protobuf主要安装protobuf的编译器,编译器目的,可以把protobuf的IDL语言,转换成具体某一种开发语言。

2、protobuf编译器安装

github上.

https://www.github.com/protocolbuffers/protobuf/releases

建议,64位,v3.19.6,是一个exe文件。

windows版本
1.直接解压缩,放置在一个特定的目录下面
2.直接配置环境变量 path
	protoc --version
	
mac版本
 brew install protobuf

3、protobuf IDEA插件

IDEA 新版本。原生支持,protocol Buffers.

  1. 2021.2版本后面的新版本,IDEA内置了Protobuf插件,和gRPC插件,都是默认勾选的。

  2. 2021.2版本之前的老版本,可以选装第三方protobuf插件。

  3. 二者不能共存

4、与Java语言相关的语法

#后续protobuf生成的java代码 一个源文件还是多个源文件 xx.java
option java_multiple_files = false; //生成一个源文件。

#指定protobuf生成的类,放置在哪个包中
option java_package = "com.suns";      //java相关

#指定的protobuf生成的外部类的名字(管理内部类【内部类才是真正开发使用】,相当于文件夹)
option java_outer_classname = "UserService";

外部类.内部类.

5、逻辑包

#对于protobuf对于文件内容的管理
package xxx;

6、导入

UserService.proto

OrderService.proto
	imort "xxx/UserService.proto"

7、protobuf 基本类型介绍

  1. 描绘RPC服务,远端接口。 Service(trift) gRpc也叫Service

  2. 结构体Struct(thrift) ; message(gRpc)

    主要是看service和message如何定义。

protobuf的类型

https://www.protobuf.dev/programing-guides/proto3/

Scalar Value Types

A scalar message field can have one of the following types – the table shows the type specified in the .proto file, and the corresponding type in the automatically generated class:

.proto TypeNotesJava/Kotlin Type[1]
doubledouble
floatfloat
int32Uses variable-length encoding. Inefficient for encoding negative numbers – if your field is likely to have negative values, use sint32 instead.int
int64Uses variable-length encoding. Inefficient for encoding negative numbers – if your field is likely to have negative values, use sint64 instead.long
uint32Uses variable-length encoding.int[2]
uint64Uses variable-length encoding.long[2]
sint32Uses variable-length encoding. Signed int value. These more efficiently encode negative numbers than regular int32s.int
sint64Uses variable-length encoding. Signed int value. These more efficiently encode negative numbers than regular int64s.long
fixed32Always four bytes. More efficient than uint32 if values are often greater than 228.int[2]
fixed64Always eight bytes. More efficient than uint64 if values are often greater than 256.long[2]
sfixed32Always four bytes.int
sfixed64Always eight bytes.long
boolboolean
stringA string must always contain UTF-8 encoded or 7-bit ASCII text, and cannot be longer than 232.String
bytesMay contain any arbitrary sequence of bytes no longer than 232.ByteString

8、枚举

enum SEASON{
	SPRING = 0;
	SUMMER = 1;

}

枚举的值 必须是0开始。

9、消息Message

9.1、基本定义

message LoginRequest{
	String username = 1;//代表编号。不是value
	singular String password = 2;
	int32 age = 3 ;
}

9.2、编号问题:

从1开始,到2^29-1,注意19000-19999 这几个编号不能用。这是protobuf自己保留的区间。

消息

- singular : 修饰messgae字段,这个字段只能是0个或1个 (默认关键字)0:null, 1:"123456"
- reapeated : 这个字段返回值是多个;等价于java List

message Result{
	string content = 1;
	reapeated string stutas = 2;  //这个字段返回值是多个;等价于java List; Protobuf, 生成一个get方法: getStatusList()->List
}

9.3、protobuf [gRpc] 可以定义多个消息

message LoginRequest{
...
}

message LoginResponse{
...
}

9.4、消息可以嵌套

message SearchResponse{
	message Result{
		string url = 1;
		string title =2;
	}
	string xxx = 1;
	string yyy = 2;
	Result ppp = 3;//内部。上面是定义message
}

其他人调用:SearchResponse.Result

message AAA{
	string xxx = 1;
	SearchResponse.Result yyy = 2;
}

9.5、oneof [其中一个]

message SimpleMessage{
	//可取值,要么是string要么是int32
	oneof test_oneof{
		string name = 1;
		int32 age =2
	}
	
	test_oneof xxx;
}

9.6、服务

service HelloService{
	rpc hello(HelloRequest) returns(HelloResponse){}
}

# 里面可以定义多个服务方法
# 定义多个服务接口
#gRpc 服务 4个服务方式.

4、第一个gRpc开发

4.1、项目结构

1. xxxx-api 模块
	定义protobuf idl语言
	并且通过命令创建具体的代码,后续client,server引入使用。
	1. message
	2. service
2. xxxx-server 模块
	1.实现api模块中定义的服务接口
	2.发布gRpc服务(创建服务器程序)
3. xxxx-client模块
	1. 创建服务端stub(代理)
	2. 基于代理(stub)RPC调用

4.2、api模块 grpc-api

1、proto文件 书写protobuf的IDL

2、protoc命令 把proto文件的IDL转换成编程语言。

#--java_out  表示 生成java  --go_out,表示生成go
#--java_out=/xxx/xxx   /xxx/xxx 表示位置
# /xxx/xxx/xx.proto  基于什么IDL的。 这个是源文件。
protoc --java_out=/xxx/xxx   /xxx/xxx/xx.proto

3、实战】maven插件 进行protobuf IDL文件的编译,并把它放置IDEA具体位置。

4、github上有开源的插件。g

http://www.github.com/grpc/grpc-java
pom.xml

    <dependencies>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-netty-shaded</artifactId>
            <version>1.57.2</version>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-protobuf</artifactId>
            <version>1.57.2</version>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-stub</artifactId>
            <version>1.57.2</version>
        </dependency>
        <dependency> <!-- necessary for Java 9+ -->
            <groupId>org.apache.tomcat</groupId>
            <artifactId>annotations-api</artifactId>
            <version>6.0.53</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>
	##生成功能
    <build>
      <extensions>
        <extension>
          <groupId>kr.motd.maven</groupId>
          <artifactId>os-maven-plugin</artifactId>
          <version>1.7.1</version>
        </extension>
      </extensions>
      <plugins>
        <plugin>
          <groupId>org.xolstice.maven.plugins</groupId>
          <artifactId>protobuf-maven-plugin</artifactId>
          <version>0.6.1</version>
          <configuration>
              ## protoc命令  生成message信息
            <protocArtifact>com.google.protobuf:protoc:3.22.3:exe:${os.detected.classifier}</protocArtifact>
            <pluginId>grpc-java</pluginId>
               ## protoc命令  生成service
            <pluginArtifact>io.grpc:protoc-gen-grpc-java:1.57.2:exe:${os.detected.classifier}</pluginArtifact>
          </configuration>
          <executions>
            <execution>
              <goals>
                <goal>compile</goal>
                <goal>compile-custom</goal>
              </goals>
            </execution>
          </executions>
        </plugin>
      </plugins>
    </build>

${os.detected.classifier} maven内置变量;

获取当前 操作系统类型 windows -x86.

goal:maven goal;

compile;

compile-custom

5、源文件如下:

syntax = "proto3";

option java_multiple_files = false;

option java_package = "com.lizy";

option java_outer_classname = "HelloProto";

/*
  命名Hello.proto,outer_classname = HelloProto

  IDL文件  目的  发布RPC服务,service---> message   message<---------


先定义message
 */

message HelloRequest{
  string name = 1;
}

message HelloResponse{
  string result = 1;
}

/*
  再定义service  需要让  protobuf转换 IDL语言。
 */

service HelloService{
  rpc hello(HelloRequest) returns (HelloResponse){}
}

这个字体蛮好看的

霞鹜文楷

使用 no meirye ui version 3.0.0

reg save "HKCU\Control Panel" .\reset_font.reg /y

https://github.com/Tatsu-syo/noMeiryoUI

6、使用mavenHelper插件

可以使用ctrl+alt+r 快捷键导入。

7、使用maven的plugins中protobuf-maven-plugin插件

生成的文件会放在target里面。

  • 需要执行两个命令

  • compile

  • compile-custom

  • 然后再去手工移动生成的文件

过于麻烦。

8、从实战出发。

问题1:需要执行两个命令

解决:maven new goal,生成一个合并的命令

mvn xxx xxx;

Run maven goal;

new maven goal
点击需要的goal,点击两个,就可以直接有两个一起。

问题2:手工移动生成的文件太过麻烦

解决:在pom.xml文件中增加两行outputDirectory,clearOutputDirectory

                <configuration>
<!--                    ## protoc命令  生成message信息-->
                    <protocArtifact>com.google.protobuf:protoc:3.22.3:exe:${os.detected.classifier}</protocArtifact>
                    <pluginId>grpc-java</pluginId>
<!--                    ## protoc命令  生成service-->
                    <pluginArtifact>io.grpc:protoc-gen-grpc-java:1.57.2:exe:${os.detected.classifier}</pluginArtifact>
                        <outputDirectory>${basedir}/src/main/java</outputDirectory>
                    <!-- false 追加式生成-->
                    <clearOutputDirectory>false</clearOutputDirectory>
                </configuration>

image-20230906214402155

ctr+o 重写父类方法。

在 gRPC 中,BlockingStub, FutureStub, 和 Stub 是用于与 gRPC 服务进行交互的不同接口。它们的主要区别在于它们提供的方法调用的方式。

BlockingStub:
BlockingStub 是 gRPC 提供的阻塞式接口。使用此接口,调用远程服务是阻塞的,也就是说,调用者必须等待服务器的响应才能继续执行。在调用方法后,你需要使用 .get() 方法来阻塞并等待结果。这是一个非常直接且易于理解的接口,但需要调用者等待,可能不适合处理大量并发的请求。

例如:

java
BlockingGrpc.newBlockingStub(channel).someRpc(request).get();
FutureStub:
FutureStub 也是阻塞式的接口,但与 BlockingStub 不同的是,它返回一个 Future 对象,这个对象表示一个尚未完成的计算或操作。你可以使用 Future.get() 来阻塞并等待结果,或者注册一个回调函数来在结果可用时自动获取。这种方式比 BlockingStub 更灵活,因为它允许你在等待结果的同时执行其他任务。

例如:

java
Futures.getUnchecked(future = asyncStub.someRpc(request));
Stub:
与前两者不同,Stub 是一个非阻塞的接口,它使用异步回调方式。当调用远程服务时,它不会立即返回结果,而是通过回调函数将结果传递给调用者。这种方式可以处理大量并发请求,并且可以更有效地利用系统资源。然而,与前两者相比,它可能需要更多的代码和更复杂的逻辑来处理异步回调。

例如:

java
channel.io().Schedulers.scheduleDirect(() -> {
    someRpc(request, new Callback() {
        @Override
        public void onSuccess(Object result) {
            // handle the result
        }

        @Override
        public void onFailure(Throwable t) {
            // handle the error
        }
    });
});
在选择使用哪种接口时,应根据你的应用需求和性能要求来决定。如果你需要简单直接的接口并且不介意阻塞等待结果,那么 BlockingStub 可能是最好的选择。如果你需要更灵活的处理方式并且愿意编写更多的代码来处理异步回调,那么 FutureStub 或 Stub 可能更适合你。

4.3、service模块 grpc-service

1、创建业务service

创建service模块,service类:HelloServiceImpl

业务端代码:

package com.lizy.service;

import com.lizy.HelloProto;
import com.lizy.HelloServiceGrpc;
import io.grpc.stub.StreamObserver;

public class HelloServiceImpl extends HelloServiceGrpc.HelloServiceImplBase {

    /**
     * 传统ssm
     * 1. 接受client的提交参数 request.getParameter()
     * 2. 业务处理 service+dao 调用对应的业务功能
     * 3. 提供返回值
     * @param request request
     * @param responseObserver responseObserver
     */
    @Override
    public void hello(HelloProto.HelloRequest request, StreamObserver<HelloProto.HelloResponse> responseObserver) {
        //1. 接受client的请求参数
        String name = request.getName();
        //2. 业务处理 todo 基本处理这一块。
        System.out.println("name parameter: "+ name);
        //3. 封装响应
        //3.1 创建响应对象的构造着
        HelloProto.HelloResponse.Builder builder = HelloProto.HelloResponse.newBuilder();
        //3.2 填充数据
        builder.setResult("hello method invoke ok");
        //3.3 封装响应
        HelloProto.HelloResponse helloResponse = builder.build();
        //处理后的响应的消息,通过网络回传 client
        responseObserver.onNext(helloResponse);
        //通知client响应已经结束 --> 标志
        responseObserver.onCompleted();
    }

}

2、创建服务端

服务端代码:

public class GrpcServer1 {
    public static void main(String[] args) throws InterruptedException, IOException {
        //1. 绑定端口
        ServerBuilder<?> serverBuilder = ServerBuilder.forPort(9000);
        //2. 发布服务
        serverBuilder.addService(new HelloServiceImpl());
        //多个服务就add多个
//        serverBuilder.addService(new UserServiceImpl());
        //3. 创建服务对象
        Server server = serverBuilder.build();

        server.start();
        server.awaitTermination();
    }
}

tips:pom.xml 父依赖 可以打印日志。

    <dependencies>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-api</artifactId>
            <version>1.7.32</version>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>1.2.9</version>
        </dependency>
    </dependencies>
   版本太高会有问题。
    

4.4、client模块 grpc-client

1、client通过代理对象完成远端对象的调用。

package com.lizy;

import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;

public class GrpcClient1 {
    public static void main(String[] args) {
        //1. 创建通信的管道
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9000).usePlaintext().build();
        //2. 获得代理对象 stub 存根。
        try {
            HelloServiceGrpc.HelloServiceBlockingStub helloService = HelloServiceGrpc.newBlockingStub(managedChannel);
            //3.完成RPC调用
            //3.1 准备参数
            HelloProto.HelloRequest.Builder builder = HelloProto.HelloRequest.newBuilder();
            builder.setName("李君临");
            HelloProto.HelloRequest helloRequest = builder.build();
            //3.2 进行功能rpc调用,获取响应的内容
            HelloProto.HelloResponse helloResponse = helloService.hello(helloRequest);
            String result = helloResponse.getResult();
            System.out.println("result = "+result);
        } catch (Exception e) {
            throw new RuntimeException(e);
        } finally {
            managedChannel.shutdown();
        }
    }
}

注意事项

client会去监听【grpc做的】 responseObserver.onCompleted();

//处理后的响应的消息,通过网络回传 client
responseObserver.onNext(helloResponse);
//通知client响应已经结束 --> 标志 ;通知client 整个服务结束,底层返回标记。
responseObserver.onCompleted();


requestObserver.onNext(helloRequest1);
requestObserver.onCompleted();

5、gRpc的四种通信方式

5.1、四种通信方式

  • 简单rpc 一元rpc(unary RPC)
  • 服务端流式RPC( Server Streaming RPC)
  • 客户端流式RPC(Client Streaming RPC)
  • 双向流RPC(Bi-directional Stream RPC)

5.2、简单RPC(一元RPC)

上面第一个RPC程序,实际上就是一元RPC

1、特点

  • 当client发起调用后,提交数据,并且等待服务端响应。
  • 开发过程中,主要采用就是一元RPC的这种通信方式。

image-20230907221021024

2、语法

service HelloService{
	rpc hello(HelloRequest) returns (HelloResponse){}
}

5.3、服务端流式RPC

一个请求对象,服务端可以回传多个结果对象。

1、特点

一个请求对象,服务端可以回传多个结果对象。

image-20230907221808218

2、使用场景

client --------> server
		股票标号
		<----------
		某一个时刻的 股票的行情

3、语法

返回值,增加stream 关键字

service HelloService{
	rpc hello(HelloRequest) returns (stream HelloResponse){}   //服务端流式RPC
	rpc hello(HelloRequest1) returns (HelloResponse1){}    //一元RPC
}

4、开发实践

4.1、代码:

1、proto增加c2ss方法;
service HelloService{
  rpc hello(HelloRequest) returns (HelloResponse){}
  rpc c2ss(HelloRequest) returns (stream HelloResponse){}
}
2、serviceImpl重写c2ss方法
    @Override
    public void c2ss(HelloProto.HelloRequest request, StreamObserver<HelloProto.HelloResponse> responseObserver) {
        //1、接收client的请求参数
        String name = request.getName();
        //2、完成业务的处理
        System.out.println("name = "+ name);
        //3、根据业务处理的结果,提供响应。

        for (int i = 0; i < 9; i++) {
            HelloProto.HelloResponse.Builder builder = HelloProto.HelloResponse.newBuilder();
            builder.setResult("处理的结果: "+ i);
            HelloProto.HelloResponse helloResponse = builder.build();
            responseObserver.onNext(helloResponse);

            //增加一个睡眠,模仿不同时刻
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
        responseObserver.onCompleted();
    }
3、client发起响应
package com.lizy;

import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;

import java.util.Iterator;

public class GrpcClient2 {
    public static void main(String[] args) {
        //1. 创建通信的管道
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9000).usePlaintext().build();
        //2. 获得代理对象 stub 存根。
        try {
            HelloServiceGrpc.HelloServiceBlockingStub helloService = HelloServiceGrpc.newBlockingStub(managedChannel);
            //3.完成RPC调用
            //3.1 准备参数
            HelloProto.HelloRequest.Builder builder = HelloProto.HelloRequest.newBuilder();
            builder.setName("李君临");
            HelloProto.HelloRequest helloRequest = builder.build();
            //3.2 进行功能rpc调用,获取响应的内容
            Iterator<HelloProto.HelloResponse> helloResponseIterator = helloService.c2ss(helloRequest);
            while (helloResponseIterator.hasNext()){
                HelloProto.HelloResponse helloResponse = helloResponseIterator.next();
                String result = helloResponse.getResult();
                System.out.println("result = "+result);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            managedChannel.shutdown();
        }
    }
}

结果示例:

image-20230907231023390

5、使用监听异步方式。

5.1、使用stub,而不是blockingstub

5.2、api,服务端代码不变

5.3、客户端代码如下

package com.lizy;

import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;
import io.grpc.stub.StreamObserver;

import java.util.concurrent.TimeUnit;

public class GrpcClient3 {
    public static void main(String[] args) {
        //1. 创建通信的管道
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9000).usePlaintext().build();
        //2. 获得代理对象 stub 存根。
        try {
            HelloServiceGrpc.HelloServiceStub helloServiceStub = HelloServiceGrpc.newStub(managedChannel);
            //3.完成RPC调用
            //3.1 准备参数
            HelloProto.HelloRequest.Builder builder = HelloProto.HelloRequest.newBuilder();
            builder.setName("李君临");
            HelloProto.HelloRequest helloRequest = builder.build();
            //3.2 进行功能rpc调用,获取响应的内容
            //StreamObserver 观察者模式
            helloServiceStub.c2ss(helloRequest, new StreamObserver<HelloProto.HelloResponse>() {
                @Override
                public void onNext(HelloProto.HelloResponse helloResponse) {
                    //服务端 响应了 一个消息后,需要立即处理话,把代码写在这里。
                    System.out.println("服务端每一次响应的消息 "+ helloResponse.getResult());
                }

                @Override
                public void onError(Throwable throwable) {

                }

                @Override
                public void onCompleted() {
                    //需要把服务端响应的所有数据拿到后,再进行业务处理。
                    //服务端 响应了 一个消息后,需要立即处理话,把代码写在这里。
                    System.out.println("服务端响应结束,后续可以根据需要,在这里统一处理服务端响应的内容");
                }
            });

            managedChannel.awaitTermination(12, TimeUnit.SECONDS);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            managedChannel.shutdown();
        }
    }
}

5.4、注意问题

异步编程,因为没有阻塞,顺序执行了代码,直接shutdown了。

服务端想响应,都没办法响应。因为client已经关掉了通道。

5.4、客户端流式RPC

1、特点

客户端发送多个请求对象,服务只返回一个结果。

image-20230914214522160

2、使用场景

IOT(互联网【传感器】)

这些终端给服务端发送数据。

传感器会经常性地给服务端发消息。

3、语法

proto

  rpc cs2s(stream HelloRequest) returns (HelloResponse){}

4、开发

4.1、api

rpc cs2s(stream HelloRequest) returns (HelloResponse){}

4.2、服务端开发

 @Override
    public StreamObserver<HelloProto.HelloRequest> cs2s(StreamObserver<HelloProto.HelloResponse> responseObserver) {
        return new StreamObserver<HelloProto.HelloRequest>() {
            @Override
            public void onNext(HelloProto.HelloRequest helloRequest) {
                System.out.println("服务端收到一条消息:"+helloRequest.getName());
            }

            @Override
            public void onError(Throwable throwable) {

            }

            @Override
            public void onCompleted() {
                System.out.println("全部接收完成");
                HelloProto.HelloResponse.Builder builder = HelloProto.HelloResponse.newBuilder();
                builder.setResult("this is result");
                HelloProto.HelloResponse helloResponse = builder.build();
                responseObserver.onNext(helloResponse);
                responseObserver.onCompleted();
            }
        };
    }

4.3、客户端开发

package com.lizy;

import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;
import io.grpc.stub.StreamObserver;

import java.util.concurrent.TimeUnit;


public class GrpcClient1 {
    public static void main(String[] args) {
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9000).usePlaintext().build();
        try {
            HelloServiceGrpc.HelloServiceStub helloServiceStub = HelloServiceGrpc.newStub(managedChannel);
            StreamObserver<HelloProto.HelloRequest> requestStreamObserver = helloServiceStub.cs2s(new StreamObserver<HelloProto.HelloResponse>() {
                @Override
                public void onNext(HelloProto.HelloResponse helloResponse) {
                    System.out.println("服务端 响应的数据内容为:" + helloResponse.getResult());
                }

                @Override
                public void onError(Throwable throwable) {

                }

                @Override
                public void onCompleted() {
                    System.out.println("服务端响应结束");
                }
            });

            for (int i = 0; i < 10; i++) {
                //对应的请求。每个请求。
                //先定义返回,在定义请求。
                HelloProto.HelloRequest.Builder builder = HelloProto.HelloRequest.newBuilder();
                builder.setName("李"+i);
                HelloProto.HelloRequest request = builder.build();
                requestStreamObserver.onNext(request);
                Thread.sleep(1000);
            }
            requestStreamObserver.onCompleted();

            managedChannel.awaitTermination(12, TimeUnit.SECONDS);
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            managedChannel.shutdown();
        }
    }
}

如果在服务端处理next返回。

会提示

警告: Cancelling the stream with status Status{code=INTERNAL, description=Too many responses, cause=null}

5.6、双向流式RPC

1、特点

客户端可以发送多个请求消息,服务端响应多个响应消息

image-20230914223319133

2、应用场景

聊天室

3、语法

rpc cs2ss(stream HelloRequest) returns (stream HelloResponse){}

4、开发

4.1、api

rpc cs2ss(stream HelloRequest) returns (stream HelloResponse){}

4.2、server

  @Override
    public StreamObserver<HelloProto.HelloRequest> cs2ss(StreamObserver<HelloProto.HelloResponse> responseObserver) {
        return new StreamObserver<HelloProto.HelloRequest>() {
            @Override
            public void onNext(HelloProto.HelloRequest helloRequest) {
                System.out.println("服务端收到一条消息:" + helloRequest.getName());
                responseObserver.onNext(HelloProto.HelloResponse.newBuilder().setResult("response "+helloRequest.getName() + "result ").build());
            }

            @Override
            public void onError(Throwable throwable) {

            }

            @Override
            public void onCompleted() {
                System.out.println("全部接收完成");
                responseObserver.onCompleted();
            }
        };
    }

4.3、client

public class GrpcClient5 {
    public static void main(String[] args) {
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9000).usePlaintext().build();
        try {
            HelloServiceGrpc.HelloServiceStub helloServiceStub = HelloServiceGrpc.newStub(managedChannel);
            StreamObserver<HelloProto.HelloRequest> requestStreamObserver = helloServiceStub.cs2ss(new StreamObserver<HelloProto.HelloResponse>() {
                @Override
                public void onNext(HelloProto.HelloResponse helloResponse) {
                    System.out.println("服务端 响应的数据内容为:" + helloResponse.getResult());
                }

                @Override
                public void onError(Throwable throwable) {

                }

                @Override
                public void onCompleted() {
                    System.out.println("服务端响应结束");
                }
            });

            for (int i = 0; i < 10; i++) {
                //对应的请求。每个请求。
                //先定义返回,在定义请求。
                HelloProto.HelloRequest.Builder builder = HelloProto.HelloRequest.newBuilder();
                builder.setName("李"+i);
                HelloProto.HelloRequest request = builder.build();
                requestStreamObserver.onNext(request);
                Thread.sleep(1000);
            }
            requestStreamObserver.onCompleted();

            managedChannel.awaitTermination(12, TimeUnit.SECONDS);
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            managedChannel.shutdown();
        }
    }
}

6、gRpc的代理方式

1. BlocingStub 
	阻塞 通信方式
2. Stub
	异步 通过监听处理的
3. FutureStub
   同步 / 异步 NettyFuture
   1. FutureStub 只能应用一元RPC
   
   

6.1、futureStub代码

package com.lizy;

import com.google.common.util.concurrent.FutureCallback;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import io.grpc.ManagedChannel;
import io.grpc.ManagedChannelBuilder;
import io.grpc.stub.StreamObserver;

import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class GrpcClient6 {
    public static void main(String[] args) {
        ManagedChannel managedChannel = ManagedChannelBuilder.forAddress("localhost", 9000).usePlaintext().build();
        try {
            HelloServiceGrpc.HelloServiceFutureStub helloServiceFutureStub = HelloServiceGrpc.newFutureStub(managedChannel);
            ListenableFuture<HelloProto.HelloResponse> listenableFuture = helloServiceFutureStub.hello(HelloProto.HelloRequest.newBuilder().setName("李1").build());

            //阻塞操作。
//            HelloProto.HelloResponse helloResponse = listenableFuture.get();
//            System.out.println(helloResponse.getResult());

            //第一种异步操作
//            listenableFuture.addListener(()->{
//                System.out.println("异步的rpc响应回来了");
//            }, Executors.newCachedThreadPool());

            //第二种异步操作。
            Futures.addCallback(listenableFuture, new FutureCallback<HelloProto.HelloResponse>() {
                @Override
                public void onSuccess(HelloProto.HelloResponse result) {
                    System.out.println("异步的rpc响应回来了"+result.getResult());
                }

                @Override
                public void onFailure(Throwable t) {

                }
            }, Executors.newCachedThreadPool());

            System.out.println("后续的操作...");

            managedChannel.awaitTermination(12, TimeUnit.SECONDS);
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            managedChannel.shutdown();
        }
    }
}

7、gRpc与SpringBoot整合

7.1、gRpc与SpringBoot整合的思想

1. grp-server
2. grpc-client

7.2、解析

Springboot与gRpc整合过程中,对于服务端做了什么封装。

image-20230916221751048

7.3、搭建开发环境

缺少依赖

cannot access com.google.protobuf.GeneratedMessageV3

解决方法:在pom中添加相关依赖

<dependency>
            <groupId>com.google.protobuf</groupId>
            <artifactId>protobuf-java</artifactId>
            <version>3.5.1</version>
</dependency>

7.4、服务端开发

1、pom依赖

   <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>com.lizy</groupId>
            <artifactId>grpc-api</artifactId>
            <version>1.0-SNAPSHOT</version>
            <exclusions>
                <exclusion>
                    <artifactId>error_prone_annotations</artifactId>
                    <groupId>com.google.errorprone</groupId>
                </exclusion>
            </exclusions>
        </dependency>
        <dependency>
            <groupId>net.devh</groupId>
            <artifactId>grpc-server-spring-boot-starter</artifactId>
            <version>2.14.0.RELEASE</version>
            <exclusions>
                <exclusion>
                    <artifactId>error_prone_annotations</artifactId>
                    <groupId>com.google.errorprone</groupId>
                </exclusion>
                <exclusion>
                    <artifactId>protobuf-java</artifactId>
                    <groupId>com.google.protobuf</groupId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>

2、yaml配置

# 核心配置的 就是gRpc的端口号
spring:
  application:
    name: boot-server
# 取消 tomcat启动。netty服务可能也要这样做
  main:
    web-application-type: none
grpc:
  server:
    port: 9000

3、service配置

package com.liziyao.grpcbootsever.service;

import com.lizy.HelloProto;
import com.lizy.HelloServiceGrpc;
import io.grpc.stub.StreamObserver;
import net.devh.boot.grpc.server.service.GrpcService;

@GrpcService
public class HelloServiceImpl extends HelloServiceGrpc.HelloServiceImplBase {

    @Override
    public void hello(HelloProto.HelloRequest request, StreamObserver<HelloProto.HelloResponse> responseObserver) {
        String name = request.getName();
        System.out.println("request 请求的name:" + name);
        responseObserver.onNext(HelloProto.HelloResponse.newBuilder().setResult("恭喜你返回成功").build());
        responseObserver.onCompleted();
    }
}

7.5、客户端开发。

分析:

image-20230916225758573

1、pom依赖。

	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-web</artifactId>
		</dependency>

		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
		<dependency>
			<groupId>com.lizy</groupId>
			<artifactId>grpc-api</artifactId>
			<version>1.0-SNAPSHOT</version>
			<exclusions>
				<exclusion>
					<artifactId>error_prone_annotations</artifactId>
					<groupId>com.google.errorprone</groupId>
				</exclusion>
			</exclusions>
		</dependency>

		<dependency>
			<groupId>net.devh</groupId>
			<artifactId>grpc-client-spring-boot-starter</artifactId>
			<version>2.14.0.RELEASE</version>
			<exclusions>
				<exclusion>
					<artifactId>error_prone_annotations</artifactId>
					<groupId>com.google.errorprone</groupId>
				</exclusion>
			</exclusions>
		</dependency>
	</dependencies>

2、yaml配置

spring:
  application:
    name: boot-client

grpc:
  client:
    grpc-server:
      address: 'static://127.0.0.1:9000'
      negotiation-type: plaintext

3、client配置

package com.liziyao.grpcbootclient.controller;

import com.lizy.HelloProto;
import com.lizy.HelloServiceGrpc;
import net.devh.boot.grpc.client.inject.GrpcClient;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class HelloController {

    @GrpcClient("grpc-server")
    private HelloServiceGrpc.HelloServiceBlockingStub stub;

    @GetMapping("hello")
    public String hello(String name){
        if (StringUtils.isEmpty(name)) {
            name = "扶摇直上";
        }
        System.out.println("你好,进入了client方法");
        HelloProto.HelloResponse response = stub.hello(HelloProto.HelloRequest.newBuilder().setName(name).build());
        String result = response.getResult();
        return result;
    }
}

4、返回结果

image-20230916234319741

8、高级用法

1. 拦截器  一元拦截器
2. Stream Tracer [监听器] 流拦截器
3. Retry Policy 客户端重试
4. NameResolver                               consule | ectd | nacos
5. 负载均衡  (pick-first,轮询)
6. grpc与微服务的整合
   序列化(protobuf) Dubbo
   grpc             Dubbo  dubbo只做服务治理
   grpc             gateway
   grpc             JWT
   grpc             nacos2.0
   grpc             替换 OpenFeign
   
7. grpc http2.x  多种通信方式。能和K8S 作整合。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1016498.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【C++深入浅出】日期类的实现

目录 一. 前言 二. 日期类的框架 三. 日期类的实现 3.1 构造函数 3.2 析构函数 3.3 赋值运算符重载 3.4 关系运算符重载 3.5 日期 /- 天数 3.6 自增与自减运算符重载 3.7 日期 - 日期 四. 完整代码 一. 前言 通过前面两期类和对象的学习&#xff0c;我们已经对C的…

MFC读取obj格式文件2

前文读取了obj格式文件中顶点相关的数量信息&#xff0c;下面读取显示obj格式文件中顶点相关的坐标信息&#xff1b;使用前面的立方体obj文件&#xff1b; void CObjtest2View::OnDraw(CDC* pDC) {CObjtest2Doc* pDoc GetDocument();ASSERT_VALID(pDoc);// TODO: add draw co…

【无公网IP内网穿透】Windows搭建Web站点

什么是cpolar&#xff1f; cpolar是一个非常强大的内网穿透工具&#xff0c;开发调试的必备利器。 它可以将本地内网服务器的HTTP、HTTPS、TCP协议端口映射为公网地址端口&#xff0c;使得公网用户可以轻松访问您的内网服务器&#xff0c;无需部署至公网服务器。支持永久免费使…

redis 主存复制

1. 前言 Redis的持久化机制&#xff0c;它很好的解决了单台Redis服务器由于意外情况导致Redis服务器进程退出或者Redis服务器宕机而造成的数据丢失问题。 在一定程度上保证了数据的安全性&#xff0c;即便是服务器宕机的情况下&#xff0c;也可以保证数据的丢失非常少。 通常…

【python量化】挖掘股价中的图关系:基于图注意力网络的股价预测模型

写在前面 近些年&#xff0c;图神经网络在时间序列预测领域发挥了重要的作用。其中&#xff0c;图注意力网络&#xff08;GAT&#xff09;是一种基于注意力机制的图神经网络&#xff0c;能够捕捉图结构数据中节点之间的复杂关系&#xff0c;从而在许多领域中取得了突出的性能。…

STM32窗口看门狗 WWDG

目录 1.什么是窗口看门狗&#xff1f; 2.窗口看门狗工作原理 3.WWDG框图 4.控制寄存器&#xff08;WWDG_CR&#xff09; 5.配置寄存器&#xff08;WWDG_CFR&#xff09; 6.状态寄存器&#xff08;WWDG_SR&#xff09; 7.超时时间计算 8.窗口看门狗实验 9.独立看门狗和…

Python:web框架之Tornado的Hello World示例

一、安装Tornado pip install tornado 安装完成后会看到显示tornado的版本号。 二、编写Hello World程序 import tornado.ioloop #导入tornado包 import tornado.web class MainHandle(tornado.web.RequestHandler):def get(self): #定义请求函数self.write("He…

从零基础学习PyQt5软件app开发

常见的GUI框架的梳理 GUI&#xff0c;全称为图形⽤户界⾯&#xff0c;⼜称为图形⽤户接⼝&#xff0c;是⼀种⼈与计算机通信的界⾯显示格式。 ⽤户打开应⽤程序或⽹站时看到的第⼀个东⻄&#xff0c;并与之交互。 ⽤户界⾯通常包括许多视觉元素&#xff0c;如图标、按钮、图形…

python学习随笔3

range的使用 range()在python很常用&#xff0c;可以进行初始化和遍历等。 # range(st,ed) # [st, ed)# range(st,ed,step) # range(st, ed, step) i,i step, i 2 * step ... () < ed切片 跟range类似。 ll[st:ed:step]容器 元组 python中的元组中内容不可以进行更…

浅谈C++|运算符重载

重载原因 C 中的运算符重载是一种特性&#xff0c;允许程序员定义自定义类类型的运算符操作。通过运算符重载&#xff0c;可以对类对象执行类似于内置类型的操作&#xff0c;例如加法、减法、乘法等。 运算符重载通过定义特定的成员函数或非成员函数来实现。成员函数的运算符重…

html怎么设置按钮返回顶部

在 HTML 中&#xff0c;我们可以通过一些代码和 CSS 样式来创建一个这样的按钮。 <button onclick"topFunction()" id"myBtn">返回顶部</button> <style> #myBtn { display: none; position: fixed; bottom: 20px; right: 30px; z-inde…

高性能 Python 编译器 -- Codon

众所周知&#xff0c;Python 是一门简单易学、具有强大功能的编程语言&#xff0c;在各种用户使用统计榜单中总是名列前茅。相应地&#xff0c;围绕 Python&#xff0c;研究者开发了各种便捷工具&#xff0c;以更好的服务于这门语言。 编译器充当着高级语言与机器之间的翻译官&…

一封来自江苏省电力设计院的表扬信

近日&#xff0c;中新赛克海睿思收到了一封来自江苏省电力设计院公司&#xff08;以下简称“江苏院”&#xff09;的表扬信。 海睿思与江苏院自达成合作以来&#xff0c;双方团队经过共同努力&#xff0c;克服了项目交付过程中的诸多困难。不仅通过数据工程的整体咨询帮助江苏院…

pt26django教程

admin 后台数据库管理 django 提供了比较完善的后台管理数据库的接口&#xff0c;可供开发过程中调用和测试使用 django 会搜集所有已注册的模型类&#xff0c;为这些模型类提拱数据管理界面&#xff0c;供开发者使用 创建后台管理帐号: [rootvm mysite2]# python3 manage.…

什么是函数重载?作用是什么?如何使用?

函数重载是指在同一个作用域内&#xff0c;允许存在多个同名函数&#xff0c;但这些函数的参数列表必须不同。根据传入的参数类型、数量或顺序的不同&#xff0c;编译器可以区分调用哪个函数。 函数重载的作用主要有以下几点&#xff1a; 提高代码的可读性和可维护性&#xff…

openlayers-17-卷帘对比

实现卷帘对比功能&#xff0c;没有进一步测试版本兼容问题&#xff0c;不错从ol的官网来看&#xff0c;ol6之前的版本的示例与ol6及其之后的版本示例并不相同 ol5 示例https://openlayers.org/en/v5.3.0/examples/layer-swipe.html?qlayerswipeol6示例 https://openlayers.org…

GIS跟踪监管系统

GIS跟踪监管系统 系统架构功能模块1. 基本功能2. 仓库管理3. 物资查询 系统采用B/S架构&#xff0c;前端使用的技术为HTMLCSSJavaScript&#xff08;Leaflet、jQuery、bootstrap等&#xff09;&#xff0c;后台采用.NET框架。 系统架构 救援物资跟踪监管系统的架构如图所示&am…

Matplotlib入门

基本使用 基本用法 import matplotlib.pyplot as plt import numpy as npxnp.linspace(-1,1,50) y2*x1plt.figure()#定义一个图像窗口 plt.plot(x,y)#画&#xff08;x&#xff0c;y&#xff09;曲线 plt.show()#显示图像figure图像 import matplotlib.pyplot as plt import …

nat的基础配置(动态nat,nat server)

目录 1.静态nat 2.动态nat &#xff08;1&#xff09;配置公网地址池 &#xff08;2&#xff09;配置acl&#xff0c;匹配做nat转换的源 &#xff08;3&#xff09;将源转换为公网地址&#xff0c;其中no-pat表示不做端口转化&#xff0c;只做一对一的地址转换 3.nat ser…

《向量数据库指南》——向量数据库Milvus Cloud为什么选择开源?

开源对我们来说是一种信仰。从最早开始研发向量数据库的时候&#xff0c;我们就相信应该让更多人了解并使用优秀的技术&#xff0c;这是我们选择做开源的原因。 无论是在 AI 领域还是其他领域&#xff0c;我们希望技术不会被少数大公司垄断。在向量数据库问世之前&#xff0c;阿…