用 FieldMask 提高 C# gRpc 的服务性能
作者:My?IO 发布时间:2023-05-03 19:19:05
标签:FieldMask,C#,gRpc,服务,性能
前言:
想象一下,有一个服务提供个多个客户端调用,但不是所有客户端都需要全部的返回参数:
比如商品列表服务返回商品的所有信息,而订单服务调用商品列表服务,但它其实只需要商品的编码和名称就够了。
当然,我们可以为这个需求单独创建一个服务,但是这样不太灵活,比如又需要商品的编码和分类的时候怎么办?
但是,大而全的服务方法会导致计算和传输成本可能很高,如果我们能够了解响应中哪些字段不需要提供给调用者,从而避免进行不必要的计算和传输,这对提高服务性能通常是非常有益的。
在实现 gRPC 服务时,我们可以使用protobuf FieldMask
实现上述功能。
一.FieldMask
默认情况下,gRPC 使用 protobuf
作为其接口定义语和数据序列化协议。
FieldMask 是一个 protobuf
消息,包含一个名为 paths 的字段,用于指定用于指定读取操作返回或更新操作修改的字段:
message FieldMask {
repeated string paths = 1;
}
下面,让我们看一个例子,如何在C# gRpc
服务中使用它。
二、Demo
1.定义 .proto 文件
在 .proto 文件中定义服务和消息:
syntax = "proto3";
option csharp_namespace = "GrpcService2";
import "google/protobuf/field_mask.proto";
package greet;
// The greeting 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;
google.protobuf.FieldMask field_mask = 2;
}
// The response message containing the greetings.
message HelloReply {
string message1 = 1;
string message2 = 2;
string message3 = 3;
string message4 = 4;
string message5 = 5;
}
关键点是下面2句:
// 引用 field_mask 消息
import "google/protobuf/field_mask.proto";
//定义请求字段
google.protobuf.FieldMask field_mask = 2;
2.实现服务端
服务端代码如下,返回了5个字段:
public class GreeterService : Greeter.GreeterBase
{
private readonly ILogger<GreeterService> _logger;
public GreeterService(ILogger<GreeterService> logger)
{
_logger = logger;
}
public override Task<HelloReply> SayHello(HelloRequest request, ServerCallContext context)
{
var reply = new HelloReply
{
Message1 = "Hello " + request.Name + ",这是第1条消息",
Message2 = "Hello " + request.Name + ",这是第2条消息",
Message3 = "Hello " + request.Name + ",这是第3条消息",
Message4 = "Hello " + request.Name + ",这是第4条消息",
Message5 = "Hello " + request.Name + ",这是第5条消息"
};
return Task.FromResult(reply);
}
}
3.实现客户端
客户端代码如下:
using var channel = GrpcChannel.ForAddress("https://localhost:5001");
var client = new Greeter.GreeterClient(channel);
FieldMask fieldMask = new FieldMask();
fieldMask.Paths.AddRange(new string[] { "message2", "message4" });
var request = new HelloRequest { Name = "My IO" };
request.FieldMask = fieldMask;
var reply = await client.SayHelloAsync(request);
Console.WriteLine($@"Greeting:
{reply.Message1}
{reply.Message2}
{reply.Message3}
{reply.Message4}
{reply.Message5}
" );
传入了 FieldMask
,这里只需要 message2
、message4
字段。
运行程序,发现有问题,还是返回了所有字段:
4.修改服务端
这其实是在服务端没有判断 fieldMask,修改服务端代码:
var mergedReply = new HelloReply();
request.FieldMask.Merge(reply, mergedReply);
return Task.FromResult(mergedReply);
结论:
在本文中,我们看到了如何使用 FieldMask ,这里仅仅是控制不返回字段,大家可以自行实现其他逻辑。
来源:https://blog.51cto.com/MyIO/5063206


猜你喜欢
- 本文所述为C#实现根据指定容器和控件名字获得控件的方法,在进行C#应用程序设计时有一定的借鉴价值。分享给大家供大家参考借鉴。具体实现方法如下
- 前言最近在优化自己之前基于Spring AOP的统一响应体的实现方案。什么是统一响应体呢?在目前的前后端分离架构下,后端主要是一个RESTf
- 1. 编译错误//代码1public static void test() throws Exception {throw ne
- SpringBoot 中配置redis作为session 缓存器。 让shiro引用本文是建立在你是使用这shiro基础之上的补充内容第一种
- 1 常量定义在程序中存在大量的数据来代表程序的状态,其中有些数据在程序运行过程中值不能发生改变,这些数据在程序中被叫做常量。2 常量语法命名
- Java注解是在JDK5时引入的新特性,鉴于目前大部分框架(如spring)都使用了注解简化代码并提高编码的效率,因此掌握并深入理解注解对于
- 本文实例为大家分享了Android绘制钟表的具体代码,供大家参考,具体内容如下首先要画一个表,我们要先知道步骤如何:1、仪表盘----外面最
- 打包与运行在项目开发完成之后,可以直接用IDEA将其打包成JAR包运行,也可以打包成WAR包运行以便在多服务器、多配置环境下运行。双击cle
- 前言开发中常用到主从数据库来提高系统的性能。怎么样才能方便的实现主从读写分离呢?近日工作任务较轻,有空学习学习技术,遂来研究如果实现读写分离
- 查看代码执行mybatis的sql语句File–>Settings–>Plugins 搜索 MyBatis Log Plugin
- jwt简介冒泡排序:(Bubble Sort)是一种简单的交换排序。之所以叫做冒泡排序,因为我们可以把每个元素当成一个小气泡,根据气泡大小,
- 本文实例讲述了Java实现批量向mysql写入数据的方法。分享给大家供大家参考,具体如下:private static String use
- 降低springcloud版本,改成Hoxton.SR5就好了,再次改成Hoxton.SR12,也不报错了,很奇怪。也发现gateway版本
- Android自定义控件属性详细介绍1. reference:参考某一资源ID。
- 本文实例为大家分享了Java Swing实现扫雷源码的具体代码,供大家参考,具体内容如下先来看下效果运行时只需要创建一个GameWindow
- 面试官Q1:请问String s = new String("xyz");产生了几个对象?对于这个Java面试题,老套路
- 前言我们在上一篇文章 中以实例讲解如何定义和使用 lambda 表达式,以及与其它语言相比,lambda 表达式在 Java 中的特殊规范。
- 本文示例实现了Android退出时关闭所有Activity的功能,分享给大家供大家参考之用。具体方法如下:一般来说,在Android退出时,
- 背景众所周知,所有被打开的系统资源,比如流、文件或者Socket连接等,都需要被开发者手动关闭,否则随着程序的不断运行,资源泄露将会累积成重
- 本文实例讲述了Android实现彩信附件的添加与删除功能。分享给大家供大家参考,具体如下:添加附件在ComposeMessageActivi