GRPC使用之HelloWorld

使用grpc的好处是提供高效的序列化能力,能够跨语言进行调用。这一节我们来学习grpc的入门应用,整篇文章分成3部分:

  1. 接口定义,使用grpc的IDL,创建proto文件,编译/生成grpc文件
  2. 服务端开发,处理客户端请求,并返回响应
  3. 客户端开发,发送请求,并接收服务端响应

根据上面的拆分,我们将grpc的项目拆分为3个工程,分别是:

  1. demo-grpc-proto,用于存放.proto文件,生成jar包供其他工程使用
  2. demo-grpc-server,引用proto,创建grpc的server
  3. demo-grpc-client,引用proto,调用grpc的server端

1. 接口定义

1. 初始化项目

先通过mvn archetype:generate从quickstart生成一个极简的Maven项目

mvn archetype:generate -DarchetypeGroupId=org.apache.maven.archetypes 
    -DarchetypeArtifactId=maven-archetype-quickstart 
    -DarchetypeVersion=1.4 
    -DgroupId=org.keyniu 
    -DartifactId=DemoGrpcProto 
    -Dversion=0.1 
    -Dpackage=org.keyniu 
    -DinteractiveMode=false       

引入grpc的Maven依赖以及打包插件,插件略复杂,主要的目的是为了将grpc生成的代码直接打包进当前工程的jar,暂时忽略具体含义即可。

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>org.keyniu.proto</groupId>
    <artifactId>demo-grpc-proto</artifactId>
    <version>1.0-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>DemoGrpcProto</name>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
    </properties>

    <dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>3.8.1</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-netty-shaded</artifactId>
            <version>1.64.0</version>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-protobuf</artifactId>
            <version>1.64.0</version>
        </dependency>
        <dependency>
            <groupId>io.grpc</groupId>
            <artifactId>grpc-stub</artifactId>
            <version>1.64.0</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>
        <sourceDirectory>${basedir}/src/main/java</sourceDirectory>
        <testSourceDirectory>${basedir}/src/test/java</testSourceDirectory>

        <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>
                    <protocArtifact>com.google.protobuf:protoc:3.25.1:exe:${os.detected.classifier}</protocArtifact>
                    <pluginId>grpc-java</pluginId>
                    <pluginArtifact>io.grpc:protoc-gen-grpc-java:1.64.0:exe:${os.detected.classifier}</pluginArtifact>
                </configuration>
                <executions>
                    <execution>
                        <goals>
                            <goal>compile</goal>
                            <goal>compile-custom</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.8.1</version>
                <configuration>
                    <source>${maven.compiler.source}</source>
                    <target>${maven.compiler.target}</target>
                    <generatedSourcesDirectory>${project.build.directory}/generated-sources/protobuf/grpc-java</generatedSourcesDirectory>
                    <generatedSourcesDirectory>${project.build.directory}/generated-sources/protobuf/java</generatedSourcesDirectory>
                </configuration>
            </plugin>

            <plugin>
                <artifactId>maven-jar-plugin</artifactId>
                <version>3.2.0</version>
                <configuration>
                    <archive>
                        <addMavenDescriptor>false</addMavenDescriptor>
                    </archive>
                </configuration>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-source-plugin</artifactId>
                <version>3.2.1</version>
                <executions>
                    <execution>
                        <id>attach-sources</id>
                        <goals>
                            <goal>jar-no-fork</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
</project>

2. 编辑proto文件

proto文件定义了grpc的接口、出参、入参,这里我们定义了4个接口,分别是对应grpc basic tutorial的4种模式:

  1. 基本调用
  2. Server端Streaming
  3. Client端Streaming
  4. 双向Streaming
    文件名是helloworld.proto,路径是 src/main/proto/helloworld.proto,文件内容:
syntax = "proto3";

option java_multiple_files = true;
option java_package = "org.keyniu.grpc.generate";
//option java_outer_classname = "HelloWorldProto";
//option objc_class_prefix = "HLW";

// The greeting service definition.
service Greeter {
  rpc sayHello (HelloRequest) returns (HelloReply) {}
  rpc sayHelloClientStream (stream HelloRequest) returns (HelloReply) {}
  rpc sayHelloServerStream (HelloRequest) returns (stream HelloReply) {}
  rpc sayHelloBiStream (stream HelloRequest) returns (stream HelloReply) {}
}

// The request message containing the user's name.
message HelloRequest {
  string name = 1;
}

// The response message containing the greetings
message HelloReply {
  string message = 1;
}

3. 编译proto文件

通过Maven命令,生成grpc的请求/响应对象,以及grpc调用对象

mvn protobuf:compile protobuf:compile-custom

生成的文件结构如下在这里插入图片描述

4. 打包代码

protobuff-maven-plugin生成的代码默认在target/generated-source/protobuff/java下,该代码不会打包到当前工程的jar中,解决方案有两种

1. 换生成目录

可以将生成代码的目录放到src/main/java下,这样我们在package的时候就会包含这部分代码。通过pom.xml指定生成目录

<plugin>  
	<groupId>org.xolstice.maven.plugins</groupId>  
	<artifactId>protobuf-maven-plugin</artifactId>  
	<version>0.6.1</version>  
	<configuration>  
		<protocArtifact>com.google.protobuf:protoc:3.25.1:exe:${os.detected.classifier}</protocArtifact>  
		<pluginId>grpc-java</pluginId>  
		<pluginArtifact>io.grpc:protoc-gen-grpc-java:1.64.0:exe:${os.detected.classifier}</pluginArtifact>  
		<outputDirectory>src/main/java</outputDirectory>  
	</configuration>  
	<executions>  
		<execution>  
			<goals>  
				<goal>compile</goal>  
				<goal>compile-custom</goal>  
			</goals>  
		</execution>  
	</executions>  
</plugin>
2. 改编译插件

直接将代码生成到src/main/java下的问题是生成的代码和手动编辑的代码冲突,而且我们无法分辨哪些是生成的,哪些自己编辑的。前面我们给出的完整配置,目的就是打包的时候将generated-source种protobuff的代码包含在jar中

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <version>3.8.1</version>
    <configuration>
        <source>${maven.compiler.source}</source>
        <target>${maven.compiler.target}</target>
        <generatedSourcesDirectory>${project.build.directory}/generated-sources/protobuf/grpc-java</generatedSourcesDirectory>
        <generatedSourcesDirectory>${project.build.directory}/generated-sources/protobuf/java</generatedSourcesDirectory>
    </configuration>
</plugin>        

正常的执行mvn clean package就能完成打包,打包后的文件结构如下图所示
在这里插入图片描述

2. 服务端

1. 初始化项目

通过achetype:generate生成一个最简单的项目

mvn archetype:generate -DarchetypeGroupId=org.apache.maven.archetypes 
    -DarchetypeArtifactId=maven-archetype-quickstart 
    -DarchetypeVersion=1.4 
    -DgroupId=org.keyniu.server
    -DartifactId=demo-grpc-server 
    -Dversion=0.1 
    -Dpackage=org.keyniu.server
    -DinteractiveMode=false     

之前我们创建了接口定义的项目demo-grpc-proto,在demo-grpc-server要引入它的依赖

<dependency>
	<groupId>org.keyniu.proto</groupId>
    <artifactId>demo-grpc-proto</artifactId>
    <version>1.0-SNAPSHOT</version>
</dependency>

2. 实现Server

proto生成的时候,根据我们定义的service名称(例子中是Greeter), 自动生成了一个GreeterGrpc类
在这里插入图片描述

要实现service Greeter服务端的逻辑,需要继承Greeter.GreeterImplBase类,下面我们来看一下每个方法的具体实现

public class GreeterServer extends GreeterGrpc.GreeterImplBase {
    public void sayHello(HelloRequest request, StreamObserver<HelloReply> observer) {
        super.sayHello(request, observer);
    }
    public StreamObserver<HelloRequest> sayHelloClientStream(StreamObserver<HelloReply> responseObserver) {
        return super.sayHelloClientStream(responseObserver);
    }
    public void sayHelloServerStream(HelloRequest request, StreamObserver<HelloReply> responseObserver) {
        super.sayHelloServerStream(request, responseObserver);
    }
    public StreamObserver<HelloRequest> sayHelloBiStream(StreamObserver<HelloReply> responseObserver) {
        return super.sayHelloBiStream(responseObserver);
    }
}
1. 基本调用

基本调用的实现逻辑清晰,通过处理入参HelloRequest,生成响应HelloReply,StreamObserver.onNext返回响应,onCompleted通知框架grpc调用结束。

    public void sayHello(HelloRequest request, StreamObserver<HelloReply> observer) {
        String message = "hello " + request.getName();
        observer.onNext(HelloReply.newBuilder().setMessage(message).build());
        observer.onCompleted();
    }
2. Server端Streaming

Server端Streaming和基本调用类似,只是多个响应时要多次调用onNext,onCompleted同样是在grpc请求完整结束之后调用

    public void sayHelloServerStream(HelloRequest request, StreamObserver<HelloReply> observer) {
        String name = request.getName();
        for(char c : name.toCharArray()) {
            String message = "hello " + String.valueOf(c);
            observer.onNext(HelloReply.newBuilder().setMessage(message).build());
        }
        observer.onCompleted();
    }
3. Client端Streaming

Client端Streaming是通过返回一个StreamObserver回调对象给框架实现的,每收到一个请求框架就回调返回对象的onNext方法,客户端发送参数结束之后会调用onCompleted方法,详细的流程看代码会更清晰。

    public StreamObserver<HelloRequest> sayHelloClientStream(StreamObserver<HelloReply> observer) {
        return new StreamObserver<HelloRequest>() {
            private List<String> names = new ArrayList<String>();
            public void onNext(HelloRequest request) {
                names.add(request.getName());
            }
            public void onError(Throwable t) {
                t.printStackTrace();
            }
            public void onCompleted() {
                String message = "hello " + String.join(",", names);
                observer.onNext(HelloReply.newBuilder().setMessage(message).build());
                observer.onCompleted();
            }
        };
    }
4. 双向Streaming

我们已经看到过Client和Server端各自的Streaming实现了,双向Streaming其实就是两者的结合,应该说StreamObserver的抽象还是很实用的。

    public StreamObserver<HelloRequest> sayHelloBiStream(StreamObserver<HelloReply> observer) {
        return new StreamObserver<HelloRequest>() {
            public void onNext(HelloRequest request) {
                String message = "hello " + request.getName();
                observer.onNext(HelloReply.newBuilder().setMessage(message).build());
            }
            public void onError(Throwable t) {
                t.printStackTrace();
            }
            public void onCompleted() {
                observer.onCompleted();
            }
        };
    }

3. 启动服务

通过ServerBuilder创建服务,将我们的GreeterServer作为服务注册

public class Main {
    public static void main(String[] args) throws IOException {
        Server server = ServerBuilder.forPort(2333)
                .addService(new GreeterServer())
                .build();
        server.start();
        System.out.println("server started....");
    }
}

启动后发现server started服务就算启动完成了。
在这里插入图片描述

3. 客户端

首先是创建和grpc服务的Channel,通过ManagedChannel创建,后续的调用都是通Stub完成的,支持的Stub有两种

  1. BlockingStub,同步调用
  2. async Stub,异步调用
ManagedChannel channel = ManagedChannelBuilder.forAddress("127.0.0.1", 2333).usePlaintext().build();
GreeterGrpc.GreeterBlockingStub stub = GreeterGrpc.newBlockingStub(channel);
GreeterGrpc.GreeterStub async = GreeterGrpc.newStub(channel);

如果不是Client端不是Stream,都可以通过BlockingStub实现,我们来分别看一下

1. 基本调用

HelloRequest request = HelloRequest.newBuilder().setName("randy").build();
HelloReply reply = stub.sayHello(request);
System.out.println(reply.getMessage());

2. Server端Streaming

直接响应一个Iterator,Iterator中是服务器的多个响应值

Iterator<HelloReply> replies = stub.sayHelloServerStream(request);
while (replies.hasNext()) {
    HelloReply aReply = replies.next();
    System.out.println(aReply.getMessage());
}

3. Client端Streaming

Client端Streaming需要接着async stub完成,我们曾经看到过两个StreamObserver,在客户端的时候,客户端的StreamObserver是用来发送数据的,服务端的StreamObserver用来接收服务器数据;在服务端的时候则相反。

StreamObserver<HelloReply> serverObserver = new StreamObserver<HelloReply>() { // 接收服务端响应的回调
    public void onNext(HelloReply helloReply) {
        System.out.println(helloReply.getMessage());
    }
    public void onError(Throwable t) {
        t.printStackTrace();
    }
    public void onCompleted() {
        System.out.println("server complete");
    }
};
StreamObserver<HelloRequest> requestObserver = async.sayHelloClientStream(serverObserver); // 返回客户端StreamObserver
requestObserver.onNext(HelloRequest.newBuilder().setName("randy").build());  // 客户端StreamObserver.onNext用来发送数据
requestObserver.onNext(HelloRequest.newBuilder().setName("zhangsan").build());
requestObserver.onCompleted();

4. 双向Streaming

理解了Client端Streaming,双向Streaming就好理解了,唯一的不同是serverObserver的onNext方法会有多次回调

StreamObserver<HelloReply> serverObserver = new StreamObserver<HelloReply>() { // 接收服务端响应的回调
    public void onNext(HelloReply helloReply) {
        System.out.println(helloReply.getMessage());
    }
    public void onError(Throwable t) {
        t.printStackTrace();
    }
    public void onCompleted() {
        System.out.println("server complete");
    }
};
requestObserver = async.sayHelloBiStream(serverObserver);
requestObserver.onNext(HelloRequest.newBuilder().setName("randy").build());
requestObserver.onNext(HelloRequest.newBuilder().setName("zhangsan").build());
requestObserver.onCompleted();

A. 参考资料

  1. what is grpc
    1. https://grpc.io/docs/what-is-grpc/introduction/
  2. basic tutorial
    1. https://grpc.io/docs/languages/java/basics/
  3. proto buff overview
    1. https://protobuf.dev/overview/
  4. proto3 guide
    1. https://protobuf.dev/programming-guides/proto3/
  5. Java Generated
    1. https://protobuf.dev/reference/java/java-generated/
  6. Github
    1. https://github.com/grpc/grpc-java

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

wsl安装Linux系统到指定位置

默认情况下,wsl安装的系统,会安装到系统C盘,长期下去,很容易把C盘的空间消耗完,从而影响系统的正常运行,所以我建议是将wsl所有的系统都安装到其它磁盘中,便于维护。 1、导出镜像 通过wsl -l -v 查看当前已安装的系统版本。 导出到当前目录位置,也可以指定目录位置。 w…

CQ 社区版2.13.3 | 支持全局开启OTP登录、文本导入功能可独立控制……

又到一月一度的 CloudQuery 发版时间啦&#xff01; 本次版本更新&#xff0c;对多个模块进行了功能的优化和完善&#xff0c;比如将文本导入与 insert 权限脱离使文本导入可单独控制&#xff1b;将工具权限与权限等级脱离&#xff0c;使其能独立授权和提权&#xff1b;操作模…

【JavaWeb程序设计】JSP编程

目录 一、编写JSP页面&#xff0c;在界面上显示1-9&#xff0c;9个链接&#xff0c;单击每个链接&#xff0c;能够在另一个页面打印该数字的平方。 1. 运行截图 2. 第一个jsp页面&#xff08;index.jsp&#xff09; 3. 第二个jsp页面&#xff08;square.jsp&#xff09; 二…

Purple Pi OH 更改SDK的编译选项

本文适用于在Purple Pi OH开发板更改SDK编译选项。触觉智能的Purple Pi OH鸿蒙开源主板&#xff0c;是华为Laval官方社区主荐的一款鸿蒙开发主板。 该主板主要针对学生党&#xff0c;极客&#xff0c;工程师&#xff0c;极大降低了开源鸿蒙开发者的入门门槛&#xff0c;具有以下…

【一念发动便是行】念头,就是命运

一个个恶念累积就是负能量&#xff0c;念头就是命运&#xff0c;克除恶念&#xff0c;防范念头&#xff0c;念头都有能量&#xff0c;学圣学须内外庄严检肃&#xff0c;言语有灵 多数人的问题都是出在念头上&#xff0c;念头&#xff0c;就是自己的命运&#xff1b; 当我们对自…

12 Dockerfile详解

目录 1. Dockerfile 2. Dockerfile构建过程 2.1. Dockerfile编写规则&#xff1a; 2.2. Docker执行Dockerfile的大致流程 2.3. 总结 3. Dockerfile指令 3.1. FROM 3.2. MAINTAINER 3.3. RUN 3.4. EXPOSE 3.5. WORKDIR 3.6. USER 3.7. ENV 3.8. VOLUME 3.9. ADD …

51单片机STC89C52RC——14.1 直流电机调速

目录 目的/效果 1&#xff1a;电机转速同步LED呼吸灯 2 通过独立按键 控制直流电机转速。 一&#xff0c;STC单片机模块 二&#xff0c;直流电机 2.1 简介 2.2 驱动电路 2.2.1 大功率器件直接驱动 2.2.2 H桥驱动 正转 反转 2.2.3 ULN2003D 引脚、电路 2.3 PWM&…

大模型LLMs概述:利用大模型 (LLMs) 解决信息抽取任务

论文标题&#xff1a;Large Language Models for Generative Information Extraction: A Survey 论文链接&#xff1a;https://arxiv.org/pdf/2312.17617.pdf 论文主要探讨了大型语言模型&#xff08;LLMs&#xff09;在生成式信息抽取&#xff08;IE&#xff09;任务中的应用…

知识库系统平台管理与优化

在之前做的关于FAQ知识库问答系统中&#xff0c;总结了相关踩坑内容&#xff0c;梳理如下&#xff0c;供大家参考。系统平台的管理与优化对于企业来说至关重要&#xff0c;它不仅关乎数据处理的效率&#xff0c;还直接影响到用户体验和业务成果。本文将从系统平台管理方式、系统…

World of Warcraft [CLASSIC] Talent Tree

World of Warcraft [CLASSIC] Talent Tree 天赋树模拟器 01&#xff09;初始化整个页面&#xff0c;选择游戏职业&#xff0c;初始化3个天赋树 02&#xff09;初始化天赋树结构&#xff0c;层次为N层 03&#xff09;每层有4个技能&#xff0c;设置可显示&#xff0c;设置隐藏…

通用代码生成器模板体系,语句和语句组

通用代码生成器或者叫动词算子式通用目的代码生成器是一组使用Java编写的通用代码生成器。它们的原理基于动词算子和域对象的笛卡尔积。它们没有使用FreeMarker和或者Velocity等现成的文件式模板引擎。而是使用java语言开发了一套专门为动词算子式代码生成器使用的模板API。而其…

uniapp启动安卓模拟器mumu

mumu模拟器下载 ADB&#xff1a; android debug bridge &#xff0c; 安卓调试桥&#xff0c;是一个多功能的命令行工具&#xff0c;他使你能够与连接的安卓设备进行交互 # adb连接安卓模拟器 adb connect 127.0.0.1:port # 查看adb设备 adb deviceshubuilderx 有内置的adb&a…

使用 Git Hooks 防止敏感信息泄露

欢迎关注公众号&#xff1a;冬瓜白 在日常开发中&#xff0c;我们可能会不小心将敏感信息提交到 Git。为了防止这种情况&#xff0c;可以利用 Git Hooks 编写一个简单的脚本&#xff0c;当发现提交中包含敏感词时&#xff0c;给出提示。 以下是一个基于 pre-commit 钩子的示例…

【MindSpore学习打卡】应用实践-计算机视觉-深入解析 Vision Transformer(ViT):从原理到实践

在近年来的深度学习领域&#xff0c;Transformer模型凭借其在自然语言处理&#xff08;NLP&#xff09;中的卓越表现&#xff0c;迅速成为研究热点。尤其是基于自注意力&#xff08;Self-Attention&#xff09;机制的模型&#xff0c;更是推动了NLP的飞速发展。然而&#xff0c…

Git代码提交流程

1. 核心流程 2. 完成流程

LeetCode 196, 73, 105

目录 196. 删除重复的电子邮箱题目链接表要求知识点思路代码 73. 矩阵置零题目链接标签简单版思路代码 优化版思路代码 105. 从前序与中序遍历序列构造二叉树题目链接标签思路代码 196. 删除重复的电子邮箱 题目链接 196. 删除重复的电子邮箱 表 表Person的字段为id和email…

我遭遇的奥数难题(持续更新)

第一题 地上有四堆石子&#xff0c;石子数分别是1、9、15、31。如果每次从其中的三堆同时各取出1个&#xff0c;然后都放入第四堆中&#xff0c;那么&#xff0c;能否经过若干次操作&#xff0c;使得四堆石子的个数都相同?(如果能&#xff0c;请说明具体操作&#xff0c;不能…

【html】许多大型网页都会有一个自己的主题色

许多网站确实会选择一种或几种特定的颜色作为他们的主题色&#xff0c;这通常是为了建立品牌识别度和一致性。 主题色在网站设计中起着至关重要的作用&#xff0c;它们不仅影响网站的视觉效果&#xff0c;还能传达品牌的情感和价值观。选择适当的主题色可以增强用户的品牌记忆…

从传统到智能:工业园区消防管理开始华丽转身

一、工业园区的消防管理现状 然而&#xff0c;当我们审视当前工业园区的消防管理现状时&#xff0c;不难发现其中存在诸多不足。首先&#xff0c;消防信息的智能化程度低&#xff0c;仿佛一位年迈的守望者&#xff0c;力不从心&#xff0c;难以即时将现场的数据信息传达至指挥…

重定向与转发

转发参数不会自动包含在新的请求中。若要将参数传递给重定向地址&#xff0c;可以在服务器端显式地添加参数到重定向URL中。 在重定向URL中包含参数 import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; impor…