写点什么

java 版 gRPC 实战之一:用 proto 生成代码

作者:程序员欣宸
  • 2022 年 3 月 29 日
  • 本文字数:5644 字

    阅读完需:约 19 分钟

java版gRPC实战之一:用proto生成代码

欢迎访问我的 GitHub

这里分类和汇总了欣宸的全部原创(含配套源码):https://github.com/zq2599/blog_demos


关于《java 版 gRPC 实战》系列

  • 《java 版 gRPC 实战》是欣宸原创的面向 java 程序员的 gRPC 开发笔记,与读者们一起通过实战来学习掌握 gRPC 的各个知识点;

关于 gRPC

  1. gRPC 是一个高性能、开源和通用的 RPC 框架,面向移动和 HTTP/2 设计。目前提供 C、Java 和 Go 语言版本,分别是:grpc, grpc-java, grpc-go. 其中 C 版本支持 C, C++, Node.js, Python, Ruby, Objective-C, PHP 和 C# 支持.

  2. gRPC 基于 HTTP/2 标准设计,带来诸如双向流、流控、头部压缩、单 TCP 连接上的多复用请求等特。这些特性使得其在移动设备上表现更好,更省电和节省空间占用。

  3. 各个进程之间可以通过 gRPC 相互调用,如下图:

核心技术

  • 为了用 java 发布 gRPC 服务,我使用的是开源库 net.devh:grpc-server-spring-boot-starter

  • 在调用其他 gRPC 服务时用的是 net.devh:grpc-client-spring-boot-starter

  • 感谢该开源库的作者 Michael 大神,您的智慧的简化了 java 程序员的 gRPC 开发工作,项目地址:https://github.com/yidongnan/grpc-spring-boot-starter

本篇概览

作为系列文章的开篇,本篇要做的事情如下:


  1. 明确依赖库和开发环境

  2. 新建父工程 grpc-tutorials,今后《java 版 gRPC 实战》系列的所有源码都在这个工程中

  3. 实战用 proto 文件自动生成 java 代码

明确依赖库和开发环境

整个系列文章涉及的依赖库和开发环境的情况如下:


  1. JDK:1.8.0_281

  2. gradle:6.7.1

  3. springboot:2.3.8.RELEASE

  4. grpc:1.35.0

  5. protobuf:3.14.0

  6. grpc-server-spring-boot-starter:2.11.0.RELEASE

  7. grpc-client-spring-boot-starter:2.11.0.RELEASE

  8. 操作系统:win10 专业版

  9. IDEA:2021.1 (Ultimate Edition)

源码下载

  • 本篇实战中的完整源码可在 GitHub 下载到,地址和链接信息如下表所示(https://github.com/zq2599/blog_demos):


  • 这个 git 项目中有多个文件夹,《java 版 gRPC 实战》系列的源码在 grpc-tutorials 文件夹下,如下图红框所示:

创建《java 版 gRPC 实战》系列的父工程

  • 新建名为 grpc-tutorials 的 gradle 工程,前面提到的库及其版本都在此工程中处理好,build.gradle 内容如下:


import java.time.OffsetDateTimeimport java.time.format.DateTimeFormatter
buildscript { repositories { maven { url 'https://plugins.gradle.org/m2/' } // 如果有私服就在此配置,如果没有请注释掉 maven { url 'http://192.168.50.43:8081/repository/aliyun-proxy/' } // 阿里云 maven { url 'http://maven.aliyun.com/nexus/content/groups/public/' }
mavenCentral() } ext { // 项目版本 projectVersion = '1.0-SNAPSHOT'
// 依赖库的版本 grpcSpringBootStarterVersion = '2.11.0.RELEASE'
// grpc版本 https://github.com/grpc/grpc-java/releases grpcVersion = '1.35.0' // protobuf版本 https://github.com/protocolbuffers/protobuf/releases protobufVersion = '3.14.0' // protobuf的gradle插件版本 protobufGradlePluginVersion = '0.8.12'
// sprignboot版本 https://github.com/spring-projects/spring-boot/releases springBootVersion = '2.3.8.RELEASE' // springcloud版本 https://github.com/spring-cloud/spring-cloud-release/releases springCloudVersion = 'Hoxton.SR9' // nacos版本 https://github.com/alibaba/spring-cloud-alibaba/releases springCloudAlibabaNacosVersion = '2.2.3.RELEASE' // security版本 https://github.com/spring-projects/spring-security-oauth/releases springSecurityOAuthVersion = '2.5.0.RELEASE' }}
plugins { id 'java' id 'java-library' id 'org.springframework.boot' version "${springBootVersion}" apply false id 'io.spring.dependency-management' version '1.0.11.RELEASE' id 'net.nemerosa.versioning' version '2.14.0' id 'com.google.protobuf' version '0.8.14' id 'io.franzbecker.gradle-lombok' version '4.0.0' apply false id 'com.github.ben-manes.versions' version '0.36.0' // gradle dependencyUpdates}
// If you attempt to build without the `--scan` parameter in `gradle 6.0+` it will cause a build error that it can't find// a buildScan property to change. This avoids that problem.if (hasProperty('buildScan')) { buildScan { termsOfServiceUrl = 'https://gradle.com/terms-of-service' termsOfServiceAgree = 'yes' }}
wrapper { gradleVersion = '6.7.1'}
def buildTimeAndDate = OffsetDateTime.now()
ext { // 构建时取得当前日期和时间 buildDate = DateTimeFormatter.ISO_LOCAL_DATE.format(buildTimeAndDate) buildTime = DateTimeFormatter.ofPattern('HH:mm:ss.SSSZ').format(buildTimeAndDate) buildRevision = versioning.info.commit}
allprojects { apply plugin: 'java' apply plugin: 'idea' apply plugin: 'eclipse' apply plugin: 'io.spring.dependency-management' apply plugin: 'io.franzbecker.gradle-lombok'
compileJava { sourceCompatibility = JavaVersion.VERSION_1_8 targetCompatibility = JavaVersion.VERSION_1_8 options.encoding = 'UTF-8' }
compileJava.options*.compilerArgs = [ '-Xlint:all', '-Xlint:-processing' ]
// Copy LICENSE tasks.withType(Jar) { from(project.rootDir) { include 'LICENSE' into 'META-INF' } }
// 写入到MANIFEST.MF中的内容 jar { manifest { attributes( 'Created-By': "${System.properties['java.version']} (${System.properties['java.vendor']} ${System.properties['java.vm.version']})".toString(), 'Built-By': 'travis', 'Build-Date': buildDate, 'Build-Time': buildTime, 'Built-OS': "${System.properties['os.name']}", 'Build-Revision': buildRevision, 'Specification-Title': project.name, 'Specification-Version': projectVersion, 'Specification-Vendor': 'Will Zhao', 'Implementation-Title': project.name, 'Implementation-Version': projectVersion, 'Implementation-Vendor': 'Will Zhao' ) } }
repositories { mavenCentral()
// 如果有私服就在此配置,如果没有请注释掉 maven { url 'http://192.168.50.43:8081/repository/aliyun-proxy/' }
// 阿里云 maven { url 'http://maven.aliyun.com/nexus/content/groups/public/' }
jcenter() }
buildscript { repositories { maven { url 'https://plugins.gradle.org/m2/' } } }}
allprojects { project -> buildscript { dependencyManagement { imports { mavenBom "org.springframework.boot:spring-boot-starter-parent:${springBootVersion}" mavenBom "org.springframework.cloud:spring-cloud-dependencies:${springCloudVersion}" mavenBom "com.google.protobuf:protobuf-bom:${protobufVersion}" mavenBom "io.grpc:grpc-bom:${grpcVersion}" mavenBom "org.junit:junit-bom:5.7.0" }
dependencies { dependency 'org.projectlombok:lombok:1.16.16' dependency 'org.apache.commons:commons-lang3:3.11' dependency 'commons-collections:commons-collections:3.2.2' dependency "net.devh:grpc-server-spring-boot-starter:${grpcSpringBootStarterVersion}" dependency "net.devh:grpc-client-spring-boot-starter:${grpcSpringBootStarterVersion}" } }
ext { micrometerVersion = dependencyManagement.importedProperties['micrometer.version'] springFrameworkVersion = dependencyManagement.importedProperties['spring-framework.version'] springSecurityVersion = dependencyManagement.importedProperties['spring-security.version'] springCloudCommonsVersion = dependencyManagement.importedProperties['spring-cloud-commons.version'] } }}
group = 'com.bolingcavalry'version = projectVersion
复制代码


  • 整个系列用到的父工程已经完成,接下来可以开始编码了;

实战用 proto 文件自动生成 java 代码

  • gRPC 服务能够用不同的语言编写,其中的关键是定义服务的 proto 文件可以被生成各种语言的代码,java 也不例外,接下来一起体验;

  • 在父工程 grpc-tutorials 下新建模块,名为 grpc-lib,其 build.gradle 内容如下,可见主要是配置了 protobuf 插件,以及生成的 java 代码如何才能被 IDE 工具加入到 source path 中:


// 根据proto生成java代码的gradle插件plugins {    id 'com.google.protobuf'}
dependencies { implementation 'io.grpc:grpc-netty-shaded' implementation 'io.grpc:grpc-protobuf' implementation 'io.grpc:grpc-stub' if (JavaVersion.current().isJava9Compatible()) { // Workaround for @javax.annotation.Generated // see: https://github.com/grpc/grpc-java/issues/3633 implementation 'jakarta.annotation:jakarta.annotation-api' }}

protobuf { protoc { artifact = "com.google.protobuf:protoc:${protobufVersion}" } // 自动生成的代码放在这里 generatedFilesBaseDir = "$projectDir/src/generated" clean { delete generatedFilesBaseDir } // 生成java代码的插件 plugins { grpc { artifact = 'io.grpc:protoc-gen-grpc-java' } } generateProtoTasks { all()*.plugins { grpc {} } }}
// 对于eclipse,通过以下脚本可以将生成的代码加入source路径中,编译时会被用到eclipse { classpath { file.beforeMerged { cp -> def generatedGrpcFolder = new org.gradle.plugins.ide.eclipse.model.SourceFolder('src/generated/main/grpc', null); generatedGrpcFolder.entryAttributes['ignore_optional_problems'] = 'true'; cp.entries.add( generatedGrpcFolder ); def generatedJavaFolder = new org.gradle.plugins.ide.eclipse.model.SourceFolder('src/generated/main/java', null); generatedJavaFolder.entryAttributes['ignore_optional_problems'] = 'true'; cp.entries.add( generatedJavaFolder ); } }}
// 对于idea,通过以下脚本可以将生成的代码加入source路径中,编译时会被用到idea { module { sourceDirs += file('src/generated/main/java') sourceDirs += file('src/generated/main/grpc') generatedSourceDirs += file('src/generated/main/java') generatedSourceDirs += file('src/generated/main/grpc') }}
复制代码


  • 在 grpc-lib 模块的 src/main/proto 目录下新增名为 helloworld.proto 的文件,这里面定义了一个 gRPC 服务,里面含有一个接口,并且还有这个接口的入参和返回结果的定义:


syntax = "proto3";
option java_multiple_files = true;// 生成java代码的packageoption java_package = "com.bolingcavalry.grpctutorials.lib";option java_outer_classname = "HelloWorldProto";
// gRPC服务service Simple { // 接口定义 rpc SayHello (HelloRequest) returns (HelloReply) { }}
// 入参的数据结构message HelloRequest { string name = 1;}
// 返回结果的数据结构message HelloReply { string message = 1;}
复制代码


  • proto 文件已经做好,接下来要根据这个文件来生成 java 代码,在 grpc-tutorials 目录下执行命令 gradle grpc-lib:generateProto,即可根据 helloworld.proto 文件生成 java 代码,执行成功后会生出下图红框中的内容,这些就是 java 代码:

  • 本篇只聊如何生成上述代码,至于这些代码的用途就留到下一篇再说吧,这里只简单提一下,SimpleGrpc 里面有抽象类 SimpleImplBase,制作 gRPC 服务的时候需要继承该类,另外,如果您要远程调用 gRPC 的 sayHello 接口,就会用到 SimpleGrpc 类中的 SimpleStub 类,其余的 HelloReply、HelloRequest 这些则是入参和返回的数据结构定义;

  • 至此,java 版 gRPC 实战的准备工作就完成了,根据 proto 文件生成 java 代码的方法也掌握了,接下来的章节咱们一起尝试服务的发布和调用;

欢迎关注 InfoQ:程序员欣宸

学习路上,你不孤单,欣宸原创一路相伴...

发布于: 2022 年 03 月 29 日阅读数: 22
用户头像

搜索"程序员欣宸",一起畅游Java宇宙 2018.04.19 加入

前腾讯、前阿里员工,从事Java后台工作,对Docker和Kubernetes充满热爱,所有文章均为作者原创,个人Github:https://github.com/zq2599/blog_demos

评论

发布
暂无评论
java版gRPC实战之一:用proto生成代码_Java_程序员欣宸_InfoQ写作平台