当前位置: 首页 > news >正文

C# 中 Grpc服务端调用客户端方法

在 gRPC 中,服务端通常不直接调用客户端的方法,因为 gRPC 的设计模型是服务端提供服务,客户端调用服务。通常情况下,服务端和客户端之间是解耦的,服务端只提供服务端点,客户端通过这些端点发起请求。

不过,如果你确实需要从服务端调用客户端的某些操作,有几种方法可以考虑:

  1. 通过消息队列或事件总线:服务端可以向消息队列或事件总线发布消息,客户端订阅这些消息并执行相应的操作。这种方式实现了服务端和客户端之间的异步通信。

  2. 通过双向流(Bidirectional Streaming):gRPC 支持双向流,即服务端和客户端都可以在同一个连接中发送消息。如果服务端需要与客户端交换数据或通知,双向流是一种合适的方式。

  3. 通过回调机制:在某些场景中,服务端可以向客户端发送请求,客户端根据请求进行操作并回调服务端。这个机制比较复杂,通常需要一个中间层来协调这种通信。

以下是一个简单的双向流示例,展示了如何在 gRPC 中实现服务端和客户端之间的双向通信。

定义 gRPC 服务

首先,在 .proto 文件中定义一个双向流的服务接口。例如:

syntax = "proto3";service ChatService {rpc ChatStream(stream ChatMessage) returns (stream ChatMessage);
}message ChatMessage {string user = 1;string message = 2;
}

实现服务端

接着,在服务端实现这个服务接口:

using Grpc.Core;
using System.Collections.Concurrent;
using System.Threading.Tasks;public class ChatServiceImpl : ChatService.ChatServiceBase
{private readonly ConcurrentBag<IServerStreamWriter<ChatMessage>> _clients = new();public override async Task ChatStream(IAsyncStreamReader<ChatMessage> requestStream, IServerStreamWriter<ChatMessage> responseStream, ServerCallContext context){// Register the client stream_clients.Add(responseStream);// Handle incoming messageswhile (await requestStream.MoveNext()){var message = requestStream.Current;Console.WriteLine($"Received message from {message.User}: {message.Message}");// Broadcast message to all clientsforeach (var client in _clients){await client.WriteAsync(new ChatMessage{User = message.User,Message = message.Message});}}// Unregister the client stream when the client disconnects_clients.TryTake(out _);}
}

实现客户端

然后,在客户端实现与服务端的双向流通信:

using Grpc.Core;
using System;
using System.Threading.Tasks;public class ChatClient
{private readonly ChatService.ChatServiceClient _client;public ChatClient(ChatService.ChatServiceClient client){_client = client;}public async Task StartChatAsync(){using var call = _client.ChatStream();// Task to read incoming messagesvar readTask = Task.Run(async () =>{await foreach (var message in call.ResponseStream.ReadAllAsync()){Console.WriteLine($"Received message from {message.User}: {message.Message}");}});// Task to send outgoing messagesvar writeTask = Task.Run(async () =>{while (true){var message = Console.ReadLine();await call.RequestStream.WriteAsync(new ChatMessage { User = "Client", Message = message });}});await Task.WhenAll(readTask, writeTask);}
}

使用示例

在主程序中使用这些实现:

class Program
{static async Task Main(string[] args){var channel = new Channel("localhost:50051", ChannelCredentials.Insecure);var client = new ChatService.ChatServiceClient(channel);var chatClient = new ChatClient(client);await chatClient.StartChatAsync();}
}

通过上述示例,你可以看到服务端和客户端如何通过双向流进行通信。服务端可以向所有连接的客户端广播消息,而客户端可以向服务端发送消息。


http://www.mrgr.cn/news/1073.html

相关文章:

  • SQL-约束篇
  • Git入门 -- 区域详解
  • 《第十九章 多媒体 - 摄像头与相册》
  • 【信创】双系统下删除Windows只保留麒麟系统
  • 如何优雅的薅羊毛之Flux.1免费使用还支持中文prompt
  • 01_React简介、基础入门
  • OpenTelemetry:新一代的开源可观测性标准
  • Redis在Linux(Centos7)单机部署和集群部署
  • 【中等】 猿人学web第一届 第7题 动态字体,随风漂移
  • 华为 OLT 添加 ONT 配置 (OMCI 管理模式)
  • 清理 Conda 缓存和 Pip 缓存
  • 【hot100篇-python刷题记录】【矩阵置零】
  • 零基础学习Redis(5) -- redis单线程模型介绍
  • 第一篇:fiddler学习日记配置抓取https和http
  • 二:《Python基础语法汇总》— 条件判断与循环结构
  • 模电实验4 - 单电源直接耦合放大电路
  • 【Qt】Qt窗口 | QDockWidget 浮动窗口
  • python使用flask实现自动根据url寻找对应目录/文件/方法,实现动态路由
  • xss GAME (xss漏洞攻击1-8)
  • html文件运行后界面反馈xxx拒绝连接