用wordpress做外贸网站,网站 不备案,软件开发公司排名国内,韩雪冬网站利用 TaskCompletionSource 在 SuperSocket 中实现跨模块异步处理客户端消息
在使用 SuperSocket 构建 TCP 服务时#xff0c;我们经常会遇到这样的需求#xff1a;
服务端接收到客户端数据后#xff0c;需要将数据交给其他模块处理处理完成后再将结果返回给调用模块或客户端…利用 TaskCompletionSource 在 SuperSocket 中实现跨模块异步处理客户端消息
在使用 SuperSocket 构建 TCP 服务时我们经常会遇到这样的需求
服务端接收到客户端数据后需要将数据交给其他模块处理处理完成后再将结果返回给调用模块或客户端希望调用模块能够异步等待处理结果而不是阻塞线程
本文将通过 TaskCompletionSource 来实现这一场景并结合 SuperSocket 的异步回调机制讲解完整实现方法。1️⃣ TaskCompletionSource 基础语法
TaskCompletionSourceT简称 TCS是 .NET 中用于手动控制 Task 完成时机的工具。
1.1 基本概念
Task异步操作的表示通常由方法内部执行完成TaskCompletionSource你可以手动控制 Task 何时完成以及完成结果是什么
var tcs new TaskCompletionSourceint();
Taskint task tcs.Task;// 模拟异步事件
Task.Run(async ()
{await Task.Delay(1000); // 模拟耗时操作tcs.SetResult(42); // 手动完成 Task 并返回结果
});int result await task;
Console.WriteLine(result); // 输出 42核心思想TaskCompletionSource 就像一个“空盒子”你自己决定什么时候放入结果并“打开盒子”。1.2 TCS 常用方法方法功能SetResult(T result)设置 Task 成功完成并返回结果TrySetResult(T result)安全写法如果 Task 已完成不会抛异常SetException(Exception ex)设置 Task 异常完成SetCanceled()取消 Task
2️⃣ 在 SuperSocket 中使用 TCS
SuperSocket 的核心是事件驱动客户端数据到达时会触发 UsePackageHandler 回调。我们可以利用 TCS将“事件”转换为“可 await 的 Task”实现异步等待消息。2.1 定义等待方法
在服务端或调用模块中定义
private TaskCompletionSourceint _tcs;public Taskint WaitForNextPackageAsync()
{_tcs new TaskCompletionSourceint();return _tcs.Task; // 返回可 await 的 Task
}外部模块调用 await WaitForNextPackageAsync() 时会挂起等待直到 _tcs.SetResult(result) 被触发2.2 接收客户端消息并触发 TCS
public event FuncStringPackageInfo, Taskint OnPackageReceived;private async ValueTask HandlePackageAsync(IAppSession session, StringPackageInfo package)
{int result 0;// 调用外部模块处理消息if (OnPackageReceived ! null){result await OnPackageReceived.Invoke(package);}// 完成 TaskCompletionSource将结果返回给等待方_tcs?.TrySetResult(result);// 同时可以给客户端发送响应await session.SendAsync(Encoding.UTF8.GetBytes(result.ToString() \r\n));
}这里实现了 消息接收与处理逻辑解耦
SuperSocket 只负责接收消息外部模块处理业务逻辑调用模块异步等待处理结果2.3 外部模块处理逻辑示例
mainWindow.OnPackageReceived async (package)
{int result 0;switch (package.Key.ToUpper()){case ADD:result package.Parameters.Select(int.Parse).Sum();break;case SUB:result package.Parameters.Select(int.Parse).Aggregate((x, y) x - y);break;case MULT:result package.Parameters.Select(int.Parse).Aggregate((x, y) x * y);break;}return result; // 返回给 TaskCompletionSource
};2.4 调用模块等待结果
int result await mainWindow.WaitForNextPackageAsync();
Console.WriteLine($处理结果: {result});外部模块就像同步等待一样获得了处理结果实际上整个流程是 异步、非阻塞 的3️⃣ 支持多客户端或多条命令
如果有多个客户端或希望同时处理多条消息可以使用 队列管理 TCS
private ConcurrentQueueTaskCompletionSourceint _queue new();public Taskint WaitNextAsync()
{var tcs new TaskCompletionSourceint();_queue.Enqueue(tcs);return tcs.Task;
}private async ValueTask HandlePackageAsync(IAppSession session, StringPackageInfo package)
{int result CalculateResult(package);if (_queue.TryDequeue(out var tcs))tcs.TrySetResult(result);await session.SendAsync(Encoding.UTF8.GetBytes(result.ToString() \r\n));
}每条消息对应一个 TCS保证多客户端/多命令都能异步等待结果4️⃣ 总结
把回调变成了异步等待这个真的是太酷啦~~~~~~~
通过 TaskCompletionSource我们可以
将事件驱动转为可 await 的异步操作实现跨模块异步处理客户端消息保持服务端与业务逻辑解耦同时支持客户端响应和模块异步等待核心模式
HandlePackageAsync 触发 → 外部模块处理 → TaskCompletionSource.SetResult → 调用模块 await 获取结果这种模式非常适合 SuperSocket、SignalR、WebSocket 等异步消息驱动场景。