網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
Stream Grpc
在我們單次投遞的數(shù)據(jù)量很大的時(shí)候,比如傳輸一個(gè)二進(jìn)制文件的時(shí)候,數(shù)據(jù)包過(guò)大,會(huì)造成瞬時(shí)傳輸壓力。或者接收方接收到數(shù)據(jù)后,需要對(duì)數(shù)據(jù)做一系列的處理工作,
比如:數(shù)據(jù)過(guò)濾 -> 數(shù)據(jù)格式轉(zhuǎn)換 -> 數(shù)據(jù)求和 ,這種場(chǎng)景非常適合使用stream grpc,
Stream Grpc演示
syntax = "proto3"; package book_stream; option go_package = "/book_stream"; service HelloStreamService { rpc BookListStream(BookListStreamRequest) returns (stream BookListStreamResponse){}; rpc CreateBookStream(stream CreateBookStreamRequest) returns (CreateBookStreamResponse){} rpc FindBookByIdStream(stream FindBookByIdStreamRequest) returns (stream FindBookByIdStreamResponse){} } message BookListStreamRequest{ } message BookListStreamResponse{ BookPoint book = 1; } message CreateBookStreamRequest{ BookPoint book = 1; } message CreateBookStreamResponse{ repeated BookIdPoint idx = 1; } message FindBookByIdStreamRequest{ BookIdPoint idx = 1; } message FindBookByIdStreamResponse{ BookPoint book = 1; } message BookIdPoint{ uint64 idx = 1; } message BookPoint{ uint64 idx = 1; string name = 2; float price = 3; string author = 4; }
運(yùn)行protoc --go_out=plugins=grpc:. *.proto
生成腳手架文件
-
BookListStream
服務(wù)端流式RPC -
CreateBookStream
客戶(hù)端流式RPC -
FindBookByIdStream
雙向流式RPC
注意,這里只是用作方便演示使用,演示方法都不是線程安全的
服務(wù)端server
var port = 8888 func main() { server := grpc.NewServer() book_stream.RegisterHelloStreamServiceServer(server, new(HelloStreamServiceImpl)) lis, err := net.Listen("tcp", fmt.Sprintf(":%d", port)) if err != nil { panic(err) } if err := server.Serve(lis); err != nil { panic(err) } }
客戶(hù)端
func main() { var port = 8888 conn, err := grpc.Dial(fmt.Sprintf(":%d", port), grpc.WithInsecure()) if err != nil { panic(err) } defer conn.Close() client := book_stream.NewHelloStreamServiceClient(conn) ctx := context.Background() if err := createBookStream(ctx, client); err != nil { panic(err) } if err := printBookList(ctx, client); err != nil { panic(err) } if err := getBookListById(ctx, client); err != nil { panic(err) } }
BookListStream
服務(wù)器端流式 RPC,顯然是單向流,并代指 Server 為 Stream 而 Client 為普通 RPC 請(qǐng)求
簡(jiǎn)單來(lái)講就是客戶(hù)端發(fā)起一次普通的 RPC 請(qǐng)求,服務(wù)端通過(guò)流式響應(yīng)多次發(fā)送數(shù)據(jù)集,客戶(hù)端 Recv 接收數(shù)據(jù)集。
server端實(shí)現(xiàn)
var bookStore = map[uint64]book_stream.BookPoint{ 1: { Idx: 1, Author: "程子", Price: 9.9, Name: "游戲思維", }, 2: { Idx: 2, Author: "丁銳", Price: 9.9, Name: "活出必要的鋒芒", }, } type HelloStreamServiceImpl struct{} func (HelloStreamServiceImpl) BookListStream(_ *book_stream.BookListStreamRequest, streamServer book_stream.HelloStreamService_BookListStreamServer) error { for idx, bookPoint := range bookStore { err := streamServer.Send(&book_stream.BookListStreamResponse{Book: &book_stream.BookPoint{ Idx: idx, Name: bookPoint.Name, Price: bookPoint.GetPrice(), Author: bookPoint.Author, }}) if err != nil { return err } } return nil }
客戶(hù)端實(shí)現(xiàn)
func printBookList(ctx context.Context, client book_stream.HelloStreamServiceClient) error { req := &book_stream.BookListStreamRequest{} listStream, err := client.BookListStream(ctx, req) if err != nil { return err } for true { resp, err := listStream.Recv() if err != nil { if err == io.EOF { return nil } return err } fmt.Printf("%v\n", *resp.Book) } return nil }
CreateBookStream
客戶(hù)端流式 RPC,單向流,客戶(hù)端通過(guò)流式發(fā)起多次?RPC 請(qǐng)求給服務(wù)端,服務(wù)端發(fā)起一次響應(yīng)給客戶(hù)端
server端實(shí)現(xiàn)
func (HelloStreamServiceImpl) CreateBookStream(server book_stream.HelloStreamService_CreateBookStreamServer) error { var resList []*book_stream.BookIdPoint for { resp, err := server.Recv() if err == io.EOF { return server.SendAndClose(&book_stream.CreateBookStreamResponse{Idx: resList}) } if err != nil { return err } bookStore[resp.Book.Idx] = *resp.Book resList = append(resList, &book_stream.BookIdPoint{Idx: resp.Book.Idx}) } }
客戶(hù)端實(shí)現(xiàn)
var newBookStore = map[uint64]book_stream.BookPoint{ 3: { Idx: 3, Author: "程子1", Price: 9.9, Name: "游戲思維1", }, 4: { Idx: 4, Author: "丁銳1", Price: 9.9, Name: "活出必要的鋒芒1", }, } func createBookStream(ctx context.Context, client book_stream.HelloStreamServiceClient) error { stream, err := client.CreateBookStream(ctx) if err != nil { return err } for _, bookPoint := range newBookStore { if err := stream.Send(&book_stream.CreateBookStreamRequest{ Book: &bookPoint, }); err != nil { return err } } recv, err := stream.CloseAndRecv() if err != nil { return err } fmt.Println(recv.Idx) return nil }
stream.SendAndClose
,它是做什么用的呢?
在這段程序中,我們對(duì)每一個(gè) Recv 都進(jìn)行了處理,當(dāng)發(fā)現(xiàn)?io.EOF
?(流關(guān)閉) 后,需要將最終的響應(yīng)結(jié)果發(fā)送給客戶(hù)端,同時(shí)關(guān)閉正在另外一側(cè)等待的 Recv
stream.CloseAndRecv
?和?stream.SendAndClose
?是配套使用的流方法,
FindBookByIdStream
服務(wù)端實(shí)現(xiàn)
func (HelloStreamServiceImpl) FindBookByIdStream(streamServer book_stream.HelloStreamService_FindBookByIdStreamServer) error { for { resp, err := streamServer.Recv() if err == io.EOF { return nil } if err != nil { return err } if book, ok := bookStore[resp.Idx.Idx]; ok { if err := streamServer.Send(&book_stream.FindBookByIdStreamResponse{Book: &book}); err != nil { return err } } } }
客戶(hù)端實(shí)現(xiàn)
func getBookListById(ctx context.Context, client book_stream.HelloStreamServiceClient) error { stream, err := client.FindBookByIdStream(ctx) if err != nil { return err } var findList = []uint64{1, 2} for _, idx := range findList { err := stream.Send(&book_stream.FindBookByIdStreamRequest{Idx: &book_stream.BookIdPoint{Idx: idx}}) if err != nil { return err } recv, err := stream.Recv() if err != nil { return err } fmt.Printf("%v\n", recv.Book) } if err := stream.CloseSend(); err != nil { return err } return nil }
原文鏈接:https://juejin.cn/post/7110794610653265933
相關(guān)推薦
- 2022-12-07 React?Refs?的使用forwardRef?源碼示例解析_React
- 2022-08-12 Python打包成exe文件的詳細(xì)操作指南_python
- 2024-01-06 RocketMQ重復(fù)消費(fèi)問(wèn)題
- 2023-01-15 Python?pytest.main()運(yùn)行測(cè)試用例_python
- 2022-04-30 Matlab繪制散點(diǎn)密度圖的教程詳解_C 語(yǔ)言
- 2022-05-28 Oracle數(shù)據(jù)庫(kù)中通用的函數(shù)實(shí)例詳解_oracle
- 2022-09-14 Redis核心原理詳細(xì)解說(shuō)_Redis
- 2022-04-14 Python函數(shù)式編程實(shí)現(xiàn)登錄注冊(cè)功能_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過(guò)濾器
- Spring Security概述快速入門(mén)
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支