gRPC 是一种与语言无关的高性能远程过程调用 (RPC) 框架。
在 gRPC 中,客户端应用程序能够直接调用不一样计算机上的服务器应用程序上的方法,就像它是本地对象同样,从而更轻松地建立分布式应用程序和服务。它基于定义服务的想法,指定了参数和返回类型的远程过程调用的方法。服务器端实现这个接口并运行grpc服务来处理客户端的请求,客户端调用相同的方法完成请求。
git
grpc官网
.NET 上的 gRPC 的简介github
gRPC 的主要优势是:web
现代高性能轻量级 RPC 框架。
协定优先 API 开发,默认使用协议缓冲区(Protocol Buffers),容许与语言无关的实现。
可用于多种语言的工具,以生成强类型服务器和客户端。
支持客户端、服务器和双向流式处理调用。
使用 Protobuf 二进制序列化减小对网络的使用。json
这些优势使 gRPC 适用于:api
效率相当重要的轻量级微服务。
须要多种语言用于开发的 Polyglot 系统。
须要处理流式处理请求或响应的点对点实时服务。
低延迟、高度可扩展的分布式系统。
开发与云服务器通讯的移动客户端。
设计一个须要准确、高效且独立于语言的新协议。
分层设计,以启用扩展,例如。身份验证、负载平衡、日志记录和监控等服务器
protocol-buffers详细介绍
在C#中会生成一个名为FirstMessage的类,基本格式以下:
first.proto网络
syntax="proto3"; //指定协议版本 package my.project;//C# namespace MyProject option csharp_namespace="GrpcDemo.Protos"; //生成C#代码时命名空间 message FirstMessage{ int32 id=1; string name=2; bool is_male=3; }
指定输入HelloRequest和输出HelloReply,以及方法名SayHello
C#会生成对应的类和方法。框架
// The greeter service definition. service Greeter { // Sends a greeting rpc SayHello (HelloRequest) returns (HelloReply) {} } // The request message containing the user's name. message HelloRequest { string name = 1; } // The response message containing the greetings message HelloReply { string message = 1;
每一个字段都会有一个惟一的字段编号,这个很是重要。json中传递数据是以字段名为key,protobuf 是以字段编号为主,因此不要轻易变化编号。
范围 1 到 15 中的字段编号须要一个字节进行编码,范围 16 到 2047 中的字段编号须要两个字节。因此须要为使用频繁的字段编号预留字段号1到15,而且为可能添加的元素预留一点字段号。asp.net
Rpc生命周期对比async
rpc SayHello(HelloRequest) returns (HelloResponse);
rpc LotsOfReplies(HelloRequest) returns (stream HelloResponse);
rpc LotsOfGreetings(stream HelloRequest) returns (HelloResponse);
rpc BidiHello(stream HelloRequest) returns (stream HelloResponse);
能够传递特定的Rpc信息(如身份信息)。形式为键值对。
var md = new Metadata { { "username","zhangsan"}, { "role","administrator"} }; //或者 var headers = new Metadata(); headers.Add("Authorization", $"Bearer {token}");
gRPC 通道提供与指定主机和端口上的 gRPC 服务器的链接。
using var channel = GrpcChannel.ForAddress("https://localhost:5001"); var client = new EmployeeService.EmployeeServiceClient(channel);
解压后,配置环境变量Path: D:\Proto\protoc-3.14.0-win64\bin
而后cmd确认安装成功:
C:\Users\Administrator>protoc --version libprotoc 3.14.0
环境说明:
visual studio 2019 16.8.5 C:\Users\Administrator>dotnet --version 5.0.103
演示特别说明:
一、client流采用分批上传图片演示。
二、服务端流采用将list数据分批传回客户端。
新建Gprc服务项目(或普通asp.netcore web项目)
一、须要依赖如下nuget包
Grpc.AspNetCore
Grpc.Tools
Grpc.Net.Client 控制台须要
Google.Protobuf
二、而后新建 Protos文件夹,定义proto文件
syntax="proto3"; option csharp_namespace="GrpcDemo.Protos"; message Employee{ int32 id=1; int32 no=2; string firstName=3; string lastName=4; float salary=5;//薪水 } message GetByNoRequest{ int32 no=1; } message EmployeeResonse{ Employee employee=1; } message GetAllReqeust{} message AddPhotoRequest{ bytes data=1; } message AddPhotoResponse{ bool isOk=1; } message EmployeeRequest{ Employee employee=1; } service EmployeeService{ //Unary Rpc示例 rpc GetByNo(GetByNoRequest) returns(EmployeeResonse); //server streaming Rpc示例 rpc GetAll(GetAllReqeust) returns(stream EmployeeResonse); //client streaming Rpc示例 rpc AddPhoto(stream AddPhotoRequest) returns(AddPhotoResponse); //双向Rpc示例 rpc SaveAll(stream EmployeeRequest) returns(stream EmployeeResonse); }
设置proto属性,
而后编译,会生成一个服务定义类以及相关的方法。
注意:EmployeeService.EmployeeServiceBase是有Grpc组件根据proto文件生成的。
public class MyEmployeeService : EmployeeService.EmployeeServiceBase { private readonly ILogger<MyEmployeeService> _logger; public MyEmployeeService(ILogger<MyEmployeeService> logger) { _logger = logger; } /// <summary> /// Unary Rpc /// </summary> /// <param name="request"></param> /// <param name="context"></param> /// <returns></returns> public override async Task<EmployeeResonse> GetByNo(GetByNoRequest request, ServerCallContext context) { Console.WriteLine("\r\nGrpcServer即将为你演示 一元Unary Rpc"); MetadataProcess(context); var data = InmemoryData.Employees.FirstOrDefault(m => m.No == request.No); if (data != null) { return await Task.FromResult(new EmployeeResonse() { Employee = data }); } throw new Exception("异常"); } private void MetadataProcess(ServerCallContext context) { var metaData = context.RequestHeaders; foreach (var item in metaData) { _logger.LogInformation($"key:{item.Key},value:{item.Value}"); } } /// <summary> /// 服务流Server streaming Rpcs /// </summary> /// <param name="request"></param> /// <param name="responseStream"></param> /// <param name="context"></param> /// <returns></returns> public override async Task GetAll(GetAllReqeust request, IServerStreamWriter<EmployeeResonse> responseStream, ServerCallContext context) { Console.WriteLine("\r\nGrpcServer即将为你演示 服务流Server streaming Rpcs"); MetadataProcess(context); foreach (var employee in InmemoryData.Employees) { Console.WriteLine($"responseStream.Write:{employee}"); await responseStream.WriteAsync(new EmployeeResonse() { Employee = employee }); } } /// <summary> /// 客户端流Client streaming RPCs /// </summary> /// <param name="requestStream"></param> /// <param name="context"></param> /// <returns></returns> public override async Task<AddPhotoResponse> AddPhoto(IAsyncStreamReader<AddPhotoRequest> requestStream, ServerCallContext context) { Console.WriteLine("\r\nGrpcServer即将为你演示 客户端流Client streaming RPCs "); MetadataProcess(context); var data = new List<byte>(); while (await requestStream.MoveNext()) { Console.WriteLine($"Received:{requestStream.Current.Data.Length}"); data.AddRange(requestStream.Current.Data); } Console.WriteLine($"Received file with{data.Count} bytes"); return new AddPhotoResponse { IsOk = true }; } /// <summary> /// 双向流Bidirectional streaming RPCs /// </summary> /// <param name="requestStream"></param> /// <param name="responseStream"></param> /// <param name="context"></param> /// <returns></returns> public override async Task SaveAll(IAsyncStreamReader<EmployeeRequest> requestStream, IServerStreamWriter<EmployeeResonse> responseStream, ServerCallContext context) { Console.WriteLine("\r\nGrpcServer即将为你演示 双向流Bidirectional streaming RPCs"); while (await requestStream.MoveNext()) { var employee = requestStream.Current.Employee; Console.WriteLine($"requestStream.Current:{employee}"); lock (this) { InmemoryData.Employees.Add(employee); } Console.WriteLine($"responseStream.Write:{employee}"); await responseStream.WriteAsync(new EmployeeResonse() { Employee = employee }); } } } }
同上面新建Console项目,并引用如下nuget包:
Google.Protobuf
Grpc.Net.Client
Google.Protobuf
Grpc.Tools
新建protos文件夹,复制proto文件(或引用其余管理方案,如在线地址),而后编译生成解决方案:
建立通道
static async Task Main(string[] args) { using var channel = GrpcChannel.ForAddress("https://localhost:5001"); var client = new EmployeeService.EmployeeServiceClient(channel); var md = new Metadata { { "username","zhangsan"}, { "role","administrator"}, { "Authorization", $"Bearer xxxxxxxxxxxxxxxxxx" } }; Console.WriteLine("\r\nGrpcClient即将为你演示 一元Unary Rpc"); await GetByNoAsync(client, md); Console.WriteLine("\r\nGrpcClient即将为你演示 服务流Server streaming Rpcs"); await GetAll(client, md); Console.WriteLine("\r\nGrpcClient即将为你演示 客户端流Client streaming RPCs "); await AddPhoto(client,md); Console.WriteLine("\r\nGrpcClient即将为你演示 双向流Bidirectional streaming RPCs"); await SaveAll(client, md); Console.WriteLine("Press Any key Exit!"); Console.Read(); }
而后对接服务端四种服务流方式:
/// <summary> /// Unary RPC一元RPC /// </summary> static async Task GetByNoAsync(EmployeeServiceClient client, Metadata md) { //一元 var response = await client.GetByNoAsync(new GetByNoRequest() { No = 1 }, md); Console.WriteLine($"Reponse:{response}"); } /// <summary> /// server-stream /// </summary> /// <param name="client"></param> /// <param name="md"></param> /// <returns></returns> static async Task GetAll(EmployeeServiceClient client, Metadata md) { using var call = client.GetAll(new GetAllReqeust() { }); var responseStream = call.ResponseStream; while (await responseStream.MoveNext()) { Console.WriteLine(responseStream.Current.Employee); } } /// <summary> /// client-stream /// </summary> /// <param name="client"></param> /// <param name="md"></param> /// <returns></returns> static async Task AddPhoto(EmployeeServiceClient client, Metadata md) { FileStream fs = File.OpenRead("Q1.png"); using var call = client.AddPhoto(md); var stram = call.RequestStream; while (true) { byte[] buffer = new byte[1024]; int numRead = await fs.ReadAsync(buffer, 0, buffer.Length); if (numRead == 0) { break; } if (numRead < buffer.Length) { Array.Resize(ref buffer, numRead); } await stram.WriteAsync(new AddPhotoRequest() { Data = ByteString.CopyFrom(buffer) }); } await stram.CompleteAsync(); var res = await call.ResponseAsync; Console.WriteLine(res.IsOk); } /// <summary> /// 双向流 /// </summary> /// <param name="client"></param> /// <param name="md"></param> /// <returns></returns> static async Task SaveAll(EmployeeServiceClient client, Metadata md) { var employees = new List<Employee>() { new Employee(){ Id=10, FirstName="F10", LastName="L10", No=10, Salary=10 }, new Employee(){ Id=11, FirstName="F11", LastName="L11", No=11, Salary=11 }, new Employee(){ Id=12, FirstName="F12", LastName="L12", No=12, Salary=12 }, }; using var call = client.SaveAll(md); var requestStream = call.RequestStream; var responseStream = call.ResponseStream; var responseTask = Task.Run(async () => { while (await responseStream.MoveNext()) { Console.WriteLine($"response:{responseStream.Current.Employee}"); } }); foreach (var employee in employees) { await requestStream.WriteAsync(new EmployeeRequest() { Employee = employee }); } await requestStream.CompleteAsync(); await responseTask; }
客户端:
GrpcClient即将为你演示 一元Unary Rpc Reponse:{ "employee": { "id": 1, "no": 1, "firstName": "FN1", "lastName": "LN1", "salary": 1 } } GrpcClient即将为你演示 服务流Server streaming Rpcs { "id": 1, "no": 1, "firstName": "FN1", "lastName": "LN1", "salary": 1 } { "id": 2, "no": 2, "firstName": "FN1", "lastName": "LN1", "salary": 1 } { "id": 3, "no": 3, "firstName": "FN1", "lastName": "LN1", "salary": 1 } { "id": 4, "no": 4, "firstName": "FN1", "lastName": "LN1", "salary": 1 } GrpcClient即将为你演示 客户端流Client streaming RPCs True GrpcClient即将为你演示 双向流Bidirectional streaming RPCs response:{ "id": 10, "no": 10, "firstName": "F10", "lastName": "L10", "salary": 10 } response:{ "id": 11, "no": 11, "firstName": "F11", "lastName": "L11", "salary": 11 } response:{ "id": 12, "no": 12, "firstName": "F12", "lastName": "L12", "salary": 12 } Press Any key Exit!
服务端输出:
GrpcServer即将为你演示 一元Unary Rpc info: GrpcDemo.Services.MyEmployeeService[0] key:authorization,value:Bearer xxxxxxxxxxxxxxxxxx info: GrpcDemo.Services.MyEmployeeService[0] key:user-agent,value:grpc-dotnet/2.35.0.0 info: GrpcDemo.Services.MyEmployeeService[0] key:username,value:zhangsan info: GrpcDemo.Services.MyEmployeeService[0] key:role,value:administrator GrpcServer即将为你演示 服务流Server streaming Rpcs info: GrpcDemo.Services.MyEmployeeService[0] key:user-agent,value:grpc-dotnet/2.35.0.0 responseStream.Write:{ "id": 1, "no": 1, "firstName": "FN1", "lastName": "LN1", "salary": 1 } responseStream.Write:{ "id": 2, "no": 2, "firstName": "FN1", "lastName": "LN1", "salary": 1 } responseStream.Write:{ "id": 3, "no": 3, "firstName": "FN1", "lastName": "LN1", "salary": 1 } responseStream.Write:{ "id": 4, "no": 4, "firstName": "FN1", "lastName": "LN1", "salary": 1 } GrpcServer即将为你演示 客户端流Client streaming RPCs info: GrpcDemo.Services.MyEmployeeService[0] key:authorization,value:Bearer xxxxxxxxxxxxxxxxxx info: GrpcDemo.Services.MyEmployeeService[0] key:user-agent,value:grpc-dotnet/2.35.0.0 info: GrpcDemo.Services.MyEmployeeService[0] key:username,value:zhangsan info: GrpcDemo.Services.MyEmployeeService[0] key:role,value:administrator Received:1024 Received:1024 Received:1024 Received:1024 Received:1024 Received:1024 Received:1024 Received:1024 Received:1024 Received:573 Received file with9789 bytes GrpcServer即将为你演示 双向流Bidirectional streaming RPCs requestStream.Current:{ "id": 10, "no": 10, "firstName": "F10", "lastName": "L10", "salary": 10 } responseStream.Write:{ "id": 10, "no": 10, "firstName": "F10", "lastName": "L10", "salary": 10 } requestStream.Current:{ "id": 11, "no": 11, "firstName": "F11", "lastName": "L11", "salary": 11 } responseStream.Write:{ "id": 11, "no": 11, "firstName": "F11", "lastName": "L11", "salary": 11 } requestStream.Current:{ "id": 12, "no": 12, "firstName": "F12", "lastName": "L12", "salary": 12 } responseStream.Write:{ "id": 12, "no": 12, "firstName": "F12", "lastName": "L12", "salary": 12 }
[从 gRPC 建立 JSON Web API](https://docs.microsoft.com/zh-cn/aspnet/core/grpc/httpapi?view=aspnetcore-5.0)
感谢观看,本篇实践到此结束。