SpringBoot+Kotlin中使用GRPC实现服务通信的示例代码
示例项目见:kotlin-grpc
一、导入依赖:
import com.google.protobuf.gradle.* plugins { id("org.springframework.boot") version "2.3.1.RELEASE" id("io.spring.dependency-management") version "1.0.9.RELEASE" id("org.asciidoctor.convert") version "1.5.9.2" kotlin("jvm") version "1.6.0" kotlin("plugin.spring") version "1.6.0" id("com.google.protobuf") version "0.9.2" } repositories { mavenCentral() } group = "com.whrss.kotlin-grpc" version = "0.0.1-SNAPSHOT" java.sourceCompatibility = JavaVersion.VERSION_1_8 java.targetCompatibility = JavaVersion.VERSION_1_8 sourceSets.main { java.srcDirs("src/main/kotlin") } extra["spring-restdocs.version"] = "2.0.5.BUILD-SNAPSHOT" val snippetsDir by extra { file("build/generated-snippets") } dependencies { implementation("com.google.protobuf:protobuf-java:3.22.2") implementation("io.grpc:grpc-protobuf:1.53.0") implementation("com.google.protobuf:protobuf-kotlin:3.22.2") implementation("io.grpc:grpc-kotlin-stub:1.3.0") implementation("io.grpc:grpc-netty:1.56.1") implementation("io.grpc:grpc-all:1.56.1") } protobuf { protoc { artifact = "com.google.protobuf:protoc:3.19.4" } plugins { id("grpc") { artifact = "io.grpc:protoc-gen-grpc-java:1.40.1" } id("grpckt") { artifact = "io.grpc:protoc-gen-grpc-kotlin:1.3.0:jdk8@jar" } } // Enable Kotlin generation generateProtoTasks { all().forEach { it.plugins { id("grpc") id("grpckt") } } }}
二、设置Proto
将 proto
文件放在 src/mian/proto
目录下
syntax = "proto3"; import "google/api/annotations.proto"; option java_multiple_files = true; package hello_world.v1; service HelloWorldService{ rpc GetUserInfo (GetUserRequest) returns (GetUserReply) { option (google.api.http) = { get: "/api/v1/users" }; } } message GetUserRequest{ // 用户showId string userId = 1; } message GetUserReply{ // 用户showId string userId = 1; }
执行 ./gradlew clean build
build成功则会在 build/generated/source/proto/main
下生成对应的 grpc
、 grpckt
、 java
文件在程序中可以直接导包引入
三、Server端
写一个 service
import hello_world.v1.GetUserReply import hello_world.v1.GetUserRequest import hello_world.v1.HelloWorldServiceGrpcKt class Service : HelloWorldServiceGrpcKt.HelloWorldServiceCoroutineImplBase() { override suspend fun getUserInfo(request: GetUserRequest) : GetUserReply { println("getItemStatistics exec") return GetUserReply.newBuilder() .setUserId(request.userId) .build() } }
在 main
入口引入启动
import io.grpc.ServerBuilder fun main() { helloServer() } fun helloServer() { val helloService = Service() val server = ServerBuilder .forPort(15001) .addService(helloService) .build() Runtime.getRuntime().addShutdownHook(Thread { server.shutdown() server.awaitTermination() }) server.start() println("server start") server.awaitTermination() println("server restart") }
四、Client 端
import hello_world.v1.GetUserRequest import hello_world.v1.HelloWorldServiceGrpc import io.grpc.ManagedChannelBuilder fun main() { val channel = ManagedChannelBuilder.forAddress("localhost", 15001).usePlaintext() val stub = HelloWorldServiceGrpc.newBlockingStub(channel.build()) val response = stub.getUserInfo(GetUserRequest.newBuilder().setUserId("0").build()) println(response) }
五、一些坑
io.grpc 和 com.google的一些依赖是有关联的,如果依赖版本之间有巨大差异,是会导致运行错误的。比如我之前使用到了一个google的一个特别老的依赖:com.google.code.google-collections:google-collect:snapshot-20080530,导致了我程序运行时提示:
Exception in thread "main" java.lang.NoSuchMethodError: 'void com.google.common.base.Preconditions.checkArgument(boolean, java.lang.String, char, java.lang.Object)'
at io.grpc.Metadata$Key.validateName(Metadata.java:754)
at io.grpc.Metadata$Key.<init>(Metadata.java:762)
at io.grpc.Metadata$Key.<init>(Metadata.java:671)
at io.grpc.Metadata$AsciiKey.<init>(Metadata.java:971)
at io.grpc.Metadata$AsciiKey.<init>(Metadata.java:966)
at io.grpc.Metadata$Key.of(Metadata.java:708)
at io.grpc.Metadata$Key.of(Metadata.java:704)
at io.grpc.internal.GrpcUtil.<clinit>(GrpcUtil.java:99)
at io.grpc.netty.Utils.<clinit>(Utils.java:83)
at io.grpc.netty.UdsNettyChannelProvider.isAvailable(UdsNettyChannelProvider.java:34)
at io.grpc.ManagedChannelRegistry$ManagedChannelPriorityAccessor.isAvailable(ManagedChannelRegistry.java:211)
at io.grpc.ManagedChannelRegistry$ManagedChannelPriorityAccessor.isAvailable(ManagedChannelRegistry.java:207)
at io.grpc.ServiceProviders.loadAll(ServiceProviders.java:68)
at io.grpc.ManagedChannelRegistry.getDefaultRegistry(ManagedChannelRegistry.java:101)
at io.grpc.ManagedChannelProvider.provider(ManagedChannelProvider.java:43)
at io.grpc.ManagedChannelBuilder.forAddress(ManagedChannelBuilder.java:39)
at com.ck567.kotlingrpc.ClientKt.main(Client.kt:9)
at com.ck567.kotlingrpc.ClientKt.main(Client.kt)
在google是发现不了具体的问题的,可以注意一下。
上面的依赖版本都是基于对应spring boot和 kotlin 版本的,如果你的版本不适配,可能也需要折腾一下,但问题应该不是很大。
到此这篇关于SpringBoot+Kotlin中使用GRPC实现服务通信的示例代码的文章就介绍到这了,更多相关SpringBoot Kotlin GRPC服务通信内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
详解MyBatis resultType与resultMap中的几种返回类型
本文主要介绍了MyBatis resultType与resultMap中的几种返回类型,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下2021-09-09在Spring MVC中使用@ControllerAdvice创建全局异常处理器的方法
在Spring MVC中,可以使用@ControllerAdvice或@RestControllerAdvice注解来定义全局异常处理器类,并使用 @ExceptionHandler注解来定义处理特定异常的方法,本文就给大家介绍了Spring MVC @ControllerAdvice创建处理器的方法,需要的朋友可以参考下2023-08-08java用LocalDateTime类获取当天时间、前一天时间及本周/本月的开始和结束时间
这篇文章主要给大家介绍了关于java使用LocalDateTime类获取当天时间、前一天时间及本周/本月的开始和结束时间的相关资料,文中通过代码示例介绍的非常详细,需要的朋友可以参考下2023-08-08IntelliJ Plugin 开发之添加第三方jar的示例代码
这篇文章主要介绍了IntelliJ Plugin 开发之添加第三方jar的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2020-09-09
最新评论