GRPC與.net core

来源:https://www.cnblogs.com/chenyishi/archive/2019/07/06/11143375.html
-Advertisement-
Play Games

系列章節 GRPC與.net core GRPC截止時間與元數據 GRPC與netcore Identity GRPC與netcore IdentityServer4 概述 GRPC的數據交互模式有: 1.單項RPC,最簡單的數據交換方式,客戶端發出單個請求,收到單個響應 2.服務端流式RPC,是在 ...


系列章節

GRPC與.net core

GRPC截止時間與元數據

GRPC與netcore Identity

GRPC與netcore IdentityServer4

概述

GRPC的數據交互模式有:

1.單項RPC,最簡單的數據交換方式,客戶端發出單個請求,收到單個響應

2.服務端流式RPC,是在服務端收到客戶端的請求之後,返回一個應答流,客戶端收到流之後處理。

3.客戶端流式RPC,與單項類似,但客戶端發送的是流式RPC

4.雙向流式RPC,調用由客戶端調用方法來初始化,而服務端則接收到客戶端的元數據,方法名和截止時間。服務端可以選擇發送回它的初始元數據或等待客戶端發送請求。下一步怎樣發展取決於應用,因為客戶端和服務端能在任意順序上讀寫 - 這些流的操作是完全獨立的。例如服務端可以一直等直到它接收到所有客戶端的消息才寫應答,或者服務端和客戶端可以像"乒乓球"一樣:服務端後得到一個請求就回送一個應答,接著客戶端根據應答來發送另一個請求,以此類推。

單項RPC較簡單不做示例了。

首先在vs2019中net core3.0中新建GRPC項目。然後定義響應的proto文件,根據proto文件生成響應的服務端與客戶端代碼。

1.服務端流式RPC

1.定義 protofile

syntax = "proto3";

option csharp_namespace = "GrpcGreeter";

package Greet;

// The greeting service definition.
service Greeter {
  // Sends a greeting
  rpc SayHello (HelloRequest) returns (HelloReply) {}
  rpc GetStreamContent (StreamRequest) returns (stream StreamContent) {}
}

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

// The response message containing the greetings.
message HelloReply {
  string message = 1;
}
message StreamRequest {
  string fileName = 1;
}
message StreamContent {
  bytes content = 1;
}

2.實現服務端Service

重新生成項目,然後實現GetStreamContent,簡單的讀取文件內容,並將內容返回給Client

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading.Tasks;
using Google.Protobuf;
using Grpc.Core;

namespace GrpcGreeter
{
    public class GreeterService : Greeter.GreeterBase
    {
        public override Task<HelloReply> SayHello(HelloRequest request, ServerCallContext context)
        {
            return Task.FromResult(new HelloReply
            {
                Message = "Hello " + request.Name
            });
        }

        public override Task GetStreamContent(StreamRequest request, IServerStreamWriter<StreamContent> responseStream, ServerCallContext context)
        {
            return Task.Run(async () =>
             {
                 using (var fs = File.Open(request.FileName, FileMode.Open)) // 從 request 中讀取文件名並打開文件流
                 {
                     var remainingLength = fs.Length; // 剩餘長度
                     var buff = new byte[1048576]; // 緩衝區,這裡我們設置為 1 Mb
                     while (remainingLength > 0) // 若未讀完則繼續讀取
                     {
                         var len = await fs.ReadAsync(buff); // 非同步從文件中讀取數據到緩衝區中
                         remainingLength -= len; // 剩餘長度減去剛纔實際讀取的長度

                         // 向流中寫入我們剛剛讀取的數據
                         await responseStream.WriteAsync(new StreamContent
                         {
                             Content = ByteString.CopyFrom(buff, 0, len)
                         });
                     }
                 }
             });
        }
    }
}

 

3.實現Client

新建一個netcore 3.0的Console項目,並引入Nuget包

Install-Package Grpc.Net.Client -Version 0.1.22-pre1
Install-Package Google.Protobuf -Version 3.8.0
Install-Package Grpc.Tools -Version 1.22.0

 

編輯項目文件,修改如下節點

<Protobuf Include="Protos\greet.proto" GrpcServices="Client" />

 

重新生成項目,Client端主要實現發送請求,請求是一個伺服器端的文件路徑。然後實現接收服務端的流,並保存到Client本地。

using Grpc.Net.Client;
using GrpcGreeter;
using System;
using System.Collections.Generic;
using System.IO;
using System.Net.Http;

namespace GrpcGreeterClient
{
    class Program
    {
        static async System.Threading.Tasks.Task Main(string[] args)
        {
            AppContext.SetSwitch(
                "System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport",
                true);
            var httpClient = new HttpClient();
            // The port number(50051) must match the port of the gRPC server.
            httpClient.BaseAddress = new Uri("http://localhost:50051");
            var client = GrpcClient.Create<Greeter.GreeterClient>(httpClient);
            //
            var reply = await client.SayHelloAsync(
                              new HelloRequest { Name = "GreeterClient" });
            Console.WriteLine("Greeting: " + reply.Message);
            Console.ReadKey();

            //
            var result = client.GetStreamContent(new StreamRequest { FileName = @"D:\Docs.zip" });  //發送請求
            var iter = result.ResponseStream;
            using (var fs = new FileStream(@"D:\Docs2.zip", FileMode.Create)) // 新建一個文件流用於存放我們獲取到數據
            {
                while (await iter.MoveNext()) // 迭代
                {
                    iter.Current.Content.WriteTo(fs); // 將數據寫入到文件流中
                }
            }

            Console.ReadKey();
        }
    }
}

 

文件生成成功

 2.客戶端流式RPC

1.定義 protofile

syntax = "proto3";

option csharp_namespace = "GRPC.TEST";

package Greet;

// The greeting service definition.
service Greeter {
  rpc getResult (stream Value) returns (Result) {}
}


//定義Value消息類型,用於客戶端消息
message Value {
    int32 value = 1;
}
//定義Result消息類型,包含總和,數字數量和平均值,用於服務端消息返回
message Result {
    int32 sum = 1;
    int32 cnt = 2;
    double avg = 3;
}

 

2.實現服務端Service

重新生成項目,並實現如下

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Threading.Tasks;
using Google.Protobuf;
using Grpc.Core;

namespace GRPC.TEST
{
    public class GreeterService : Greeter.GreeterBase
    {
        public override async Task<Result> getResult(IAsyncStreamReader<Value> requestStream, ServerCallContext context)
        {
            while (await requestStream.MoveNext())
            {
                var point = requestStream.Current;
            }
            return new Result { Sum = 1 };
        }
    }
}

 

3.實現Client

新建一個netcore 3.0的Console項目,並引入Nuget包,安裝nuget包與其他操作同上一個例子,實現代碼如下

using Grpc.Net.Client;
using System;
using System.Collections.Generic;
using System.IO;
using System.Net.Http;

namespace GRPC.TEST.CLIENT
{
    class Program
    {
        static async System.Threading.Tasks.Task Main(string[] args)
        {
            AppContext.SetSwitch(
                 "System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport",
                 true);
            var httpClient = new HttpClient();
            // The port number(50051) must match the port of the gRPC server.
            httpClient.BaseAddress = new Uri("http://localhost:50051");
            var client = GrpcClient.Create<Greeter.GreeterClient>(httpClient);
            using (var call = client.getResult())
            {
                await call.RequestStream.WriteAsync(new Value { Value_ = 1 });
                await call.RequestStream.CompleteAsync();
                var response = await call.ResponseAsync;
            }

            Console.ReadKey();
        }
    }
}

 

 3.雙向流式RPC

1.定義proto

syntax = "proto3";

option csharp_namespace = "GRPC.TEST";

package Greet;

// The greeting service definition.
service Greeter {
  rpc getResult (stream Value) returns (stream Result) {}
}


//定義Value消息類型,用於客戶端消息
message Value {
    int32 value = 1;
}
//定義Result消息類型,包含總和,數字數量和平均值,用於服務端消息返回
message Result {
    int32 sum = 1;
    int32 cnt = 2;
    double avg = 3;
}

 

2.服務端實現

重新生成項目,並實現如下

public override async Task getResult(IAsyncStreamReader<Value> requestStream, IServerStreamWriter<Result> responseStream, ServerCallContext context)
        {
            while (await requestStream.MoveNext())
            {
                var note = requestStream.Current;

                await responseStream.WriteAsync(new Result { Sum = 100 });
            }
        }

 

3.客戶端代碼

新建一個netcore 3.0的Console項目,並引入Nuget包,安裝nuget包與其他操作同上一個例子,實現代碼如下

using Grpc.Net.Client;
using System;
using System.Collections.Generic;
using System.IO;
using System.Net.Http;
using System.Threading.Tasks;

namespace GRPC.TEST.CLIENT
{
    class Program
    {
        static async System.Threading.Tasks.Task Main(string[] args)
        {
            AppContext.SetSwitch(
       "System.Net.Http.SocketsHttpHandler.Http2UnencryptedSupport",
       true);
            var httpClient = new HttpClient();
            // The port number(50051) must match the port of the gRPC server.
            httpClient.BaseAddress = new Uri("http://localhost:50051");
            var client = GrpcClient.Create<Greeter.GreeterClient>(httpClient);
            using (var call = client.getResult())
            {
                var responseReaderTask = Task.Run(async () =>
                {
                    while (await call.ResponseStream.MoveNext())
                    {
                        var note = call.ResponseStream.Current;
                        Console.WriteLine("Received " + note);
                    }
                });

                await call.RequestStream.WriteAsync(new Value { Value_ = 12 });
                await call.RequestStream.CompleteAsync();
                await responseReaderTask;
            }

            Console.ReadKey();
        }
    }
}

 

至此,GRPC的幾種數據交互分享完畢


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 9-7 管理員: 管理員是一種特殊的用戶。編寫一個名為Admin的類,並讓它繼承練習9-3或者9-5的User類。添加一個名為privileges的屬性,用於存儲一個由字元串(如"can add post"、"can delete post"、"can ban user"等)組成的列表。編寫一個名為 ...
  • 開發環境: Windows操作系統開發工具:MyEclipse/Eclipse + JDK+ Tomcat + MySQL 資料庫項目截圖: 獲取源碼請聯繫博主-Q:782827013 ...
  • 世界是物質的,物質是運動的,運動是有規律的,規律是可以被認識的 二項式反演 $$ g_n=\sum_{i=0}^n \binom{n}if_i\Rightarrow f_n=\sum_{i=0}^n( 1)^{n i}\binom{n}ig_i $$ 證明如下 $$ \begin{aligned} ...
  • 博主最近在學習加法器、乘法器、IEEE的浮點數標準,作為數字IC的基礎。當看到booth編碼的乘法器時,對booth編碼不是很理解,然後在網上找各種理解,終於豁然開朗。現將一個很好的解釋分享給大家,希望能對大家有所幫助。 首先,看看這幾個公式: 可以證明的是,這三個公式是相等的,一個有符號的二進位數 ...
  • FileInputStream : 輸入流 int available() : 一次讀取所有的位元組數 read() : 將文件上的位元組讀取到記憶體的數組中 FileOutputStream : 輸出流 write(byte[] b) : 將位元組數組中的位元組數據寫到文件上 緩衝區會在記憶體中創建一個819 ...
  • 好久沒有更新博客了,都有點對不起這個賬號了。這次跟大家分享的是一種編程思路,沒什麼技術含量,但也許能幫得到你。 我們經常會在程式程式中用到 Sleep 這個方法。Sleep 方法用起來非常簡單,但是有個問題,就是如果 Sleep 時間過長,突然想結束 Sleep,似乎沒有什麼好的辦法,至少目前我是不 ...
  • 在非同步轉同步時,使用不當容易造成死鎖(程式卡死) 看如下案例: 有一個非同步方法 在執行如上非同步方法時,嘗試將其轉換為同步方法 按照官方文檔《使用任務簡化非同步編程》,TaskCompletionSource使用步驟: 但是,以上邏輯執行時,界面會卡死!卡死效果如下,卡死的時候點擊界面其它按鈕無任何反應 ...
  • 本文通過TaskCompletionSource,實現非同步轉同步 首先有一個非同步方法,如下非同步任務延時2秒後,返回一個結果 如何使用TaskCompletionSource將此非同步方法轉成同步呢? TaskCompletionSource使用步驟: 測試結果: 關鍵字:非同步轉同步,TaskCompl ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...