用一个例子说说gRPC的四种服务方法
本文通过一个简单的例子来演示这4种类型的使用方法
案例代码:https://github.com/artfoxe6/grpc-sample-example
目录结构说明
├── calculator.proto # 定义 protobuf
├── client
│ ├── client.go # 客户端
│ ├── gencode
│ │ └── calculator.pb.go # protoc-gen-go中间件生成的go中间代码
│ ├── go.mod
│ └── go.sum
├── README.md
└── server
├── gencode
│ └── calculator.pb.go # 和上面一样
├── go.mod
├── go.sum
└── server.go # 服务端
client和server是两个完全独立的项目,所以protoc-gen-go生成的中间代码并没有公用,
而是放到各自单独的项目中,实际上两者内容是完全一样的
首先来看 calculator.proto
syntax = "proto3"; package rpc.calculator.test; service Calculator {
//简单模式
rpc Add (TwoNum) returns (Response) {} //服务端流
rpc GetStream (TwoNum) returns (stream Response) {} //客户端流
rpc PutStream (stream OneNum) returns (Response) {} //双向流
rpc DoubleStream (stream TwoNum) returns (stream Response) {}
} message TwoNum {
int32 a = ;
int32 b = ;
}
message Response {
int32 c = ;
}
message OneNum{
int32 a = ;
}
这里定义了一个计算器服务,包含四个方法,这个四个方法分别演示了个RPC的四种服务类型
下面分别介绍这个四个方法是做什么的:
1. 客户端发送一个请求,包含两个数字,服务端是返回两个数字的和,这种最基本也是最常用的叫简单模式,
客户端发送一次,服务端返回一次,类似于接口请求
2. 客户端发送一个请求包含两个数字,服务端返回多次,第一次返回两数子和,第二次返回两数字乘,
这种叫服务端流模式,形象点说就是服务端向客户端搭了一根单向水管,可以不停的往客户端发送数据,
客户端不停的接收,直到接收到服务端发送的结束标记后停止接收
使用场景:
客户端请求一个数据列表,但是这个数据太多了,不可能一次返回,就可以利用这种模式,
服务端一次返回100条数据,前端一直接收处理
3. 客户端发送了很多次数据,数据是单个数字,服务端不停的接收数据,客户端发送结束后服务端返回所有数据的总和,
这就是客户端流模式,形象点说就是客户端往服务端搭了一个单向的水管,可以不停的往服务端发送数据,
服务端不停的接收,直到接收到客户端发送的结束标记后停止接收,处理完数据后一次性返回给客户端,
4. 客户端分多次发送数据给服务端,每次数据是两个数,服务端收到数据后多次返回给服务端,每次返回两个数之和,
客户端可以多次给服务端发送数据,服务端可以多次返回数据,这就是双向流模式,双方都需要同时处理发送数据和接收数据,
直到双方通道流关闭
利用calculator.proto生成go中间代码
在项目跟目录执行:
protoc -I ./ ./calculator.proto --go_out=plugins=grpc:./server/gencode
protoc -I ./ ./calculator.proto --go_out=plugins=grpc:./client/gencode
分别在 ./server/gencode 和 ./client/gencode 两个目录生成了相同的go中间代码
有了中间代码后就可以编写服务端server.go了,由于篇幅限制,这里就不贴完整代码了,只对几个关键点做分析
导入中间代码
import pb "rpcserver/gencode"
定义了一个server,继承自 UnimplementedCalculatorServer
type server struct {
pb.UnimplementedCalculatorServer
}
这里有个技巧,打开calculator.pb.go你会发现,
UnimplementedCalculatorServer类已经"实现"了我们在calculator.proto中定义的所有个方法
type UnimplementedCalculatorServer struct {
}
func (*UnimplementedCalculatorServer) Add(ctx context.Context, req *TwoNum) (*Response, error) {
return nil, status.Errorf(codes.Unimplemented, "method Add not implemented")
}
....
如果你是新手,可以直接粘贴到server,稍作修改就可以了,函数体还是得自己写
func (s *server) Add(ctx context.Context, in *pb.TwoNum) (*pb.Response, error) {
return &pb.Response{C: in.A + in.B}, nil
}
其实gRPC的函数不多,定义也非常清晰,比如在写下面这个方法的时候
func (s *server) GetStream(in *pb.TwoNum, pipe pb.Calculator_GetStreamServer) error {
_ = pipe.Send(&pb.Response{C: in.A + in.B})
time.Sleep(time.Second * )
_ = pipe.Send(&pb.Response{C: in.A * in.B})
return nil
}
我也不知道有Send方法,如果用的GoLand,pipe点一下,能用的函数就出来了,就几个,一看就知道什么作用,
全程都没有用到godoc,当然我还是推荐你仔细看一下文档 https://godoc.org/google.golang.org/grpc
在来看一下客户端client.go
导入中间代码
import pb "rpcclient/gencode"
客户端就不说了,看一下代码吧
用一个例子说说gRPC的四种服务方法的更多相关文章
- java gRPC四种服务类型简单示例
一.gRPC 简介 gRPC 是Go实现的:一个高性能,开源,将移动和HTTP/2放在首位通用的RPC框架.使用gRPC可以在客户端调用不同机器上的服务端的方法,而客户端和服务端的开发语言和 运行环境 ...
- gRPC学习之四:实战四类服务方法
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- C语言中返回字符串函数的四种实现方法 2015-05-17 15:00 23人阅读 评论(0) 收藏
C语言中返回字符串函数的四种实现方法 分类: UNIX/LINUX C/C++ 2010-12-29 02:54 11954人阅读 评论(1) 收藏 举报 语言func存储 有四种方式: 1.使用堆空 ...
- C语言中返回字符串函数的四种实现方法
转自C语言中返回字符串函数的四种实现方法 其实就是要返回一个有效的指针,尾部变量退出后就无效了. 有四种方式: 1.使用堆空间,返回申请的堆地址,注意释放 2.函数参数传递指针,返回该指针 3.返回函 ...
- Django的POST请求时因为开启防止csrf,报403错误,及四种解决方法
Django默认开启防止csrf(跨站点请求伪造)攻击,在post请求时,没有上传 csrf字段,导致校验失败,报403错误 解决方法1: 注释掉此段代码,即可. 缺点:导致Django项目完全无法防 ...
- 百度地图和高德地图坐标系的互相转换 四种Sandcastle方法生成c#.net帮助类帮助文档 文档API生成神器SandCastle使用心得 ASP.NET Core
百度地图和高德地图坐标系的互相转换 GPS.谷歌.百度.高德坐标相互转换 一.在进行地图开发过程中,我们一般能接触到以下三种类型的地图坐标系: 1.WGS-84原始坐标系,一般用国际GPS纪录仪记 ...
- AJPFX关于Java中运用数组的四种排序方法
JAVA中在运用数组进行排序功能时,一般有四种方法:快速排序法.冒泡法.选择排序法.插入排序法.快速排序法主要是运用了Arrays中的一个方法Arrays.sort()实现.冒泡法是运用遍历数组进行比 ...
- javascript DOM和DOM操作的四种基本方法
在了解了javascript的语言特性后,javascript真正大放光彩的地方来了——这就是javascript DOM Javascript DOM DOM(Document Object Mod ...
- c#数据四种执行方法(ExecuteNonQuery)-----转载
c#数据四种执行方法(ExecuteNonQuery) 1.使用ExecuteReader()操作数据库 2.使用ExecuteNonQuery()操作数据库 3.使用ExecuteScalar( ...
随机推荐
- almost最好的Vue + Typescript系列02 项目结构篇
基于vue-cli 3.x,配合typescript的环境构建的新vue项目,跟以前的结构相比,有了一些变化,下面我们来简单的了解一下 基本结构: node_modules: 项目中安装的依赖模块 p ...
- Redis系列三 - 缓存雪崩、击穿、穿透
前言 从学校出来,做开发工作也有一定时间了,最近有想系统地进一步深入学习,但发现基础知识不够扎实,故此来回顾基础知识,进一步巩固.加深印象. 最初开始接触编程时,总是自己跌跌撞撞.不断摸索地去学习,再 ...
- 深入理解 vertical-align 属性
语法 用来指定行内元素或表格元素的垂直对齐方式 相对父元素的值 baseline 使元素的基线与父元素的基线对齐.HTML规范没有详细说明部分可替换元素的基线,如textarea,这意味着这些元素使用 ...
- django中CBV源码分析
前言:Django的视图处理方式有两种: FBV(function base views) 是在视图里基于函数形式处理请求. CBV(class base views)是在视图里基于类的形式处理请求. ...
- Go 的 http 包的源码,通过代码我们可以看到整个的 http 处理过程
func (srv *Server) Serve(l net.Listener) error {defer l.Close() var tempDelay time.Duration // how l ...
- 【Weiss】【第03章】练习3.2
[练习3.2] 给你一个链表L和另一个链表P,它们包含以升序排列的整数.操作printlots(L,P)将打印L中那些由P所指定的位置上的元素. 例如,如果p=1,3,4,6,那么,L的第一.第三.第 ...
- python-参考书
真的是找了很多的参考书,但是看懂,上手用的就一本比较好的<编程小白的第1本python入门书>非常的好. 属于那种一看就懂,能说明白的.别的,要么就是翻译的外文的,有点难以理解,要么就是中 ...
- 使用JDBC工具类模拟登陆验证-Java(新手)
模拟登陆验证: package JdbcDome; import java.sql.Connection; import java.sql.PreparedStatement; import java ...
- Proteomic Profiling of Paired Interstitial Fluids Reveals Dysregulated Pathways and Salivary NID1 as a Biomarker of Oral Cavity Squamous Cell Carcinoma (解读人:张聪敏)
文献名:Proteomic Profiling of Paired Interstitial Fluids Reveals Dysregulated Pathways and Salivary NID ...
- Java 14 发布了,再也不怕 NullPointerException 了!
2020年3月17日发布,Java正式发布了JDK 14 ,目前已经可以开放下载.在JDK 14中,共有16个新特性,本文主要来介绍其中的一个特性:JEP 358: Helpful NullPoint ...