首页 > 解决方案 > 何以比这更好的方式创建异步 TCP 服务器?

问题描述

我有一个服务器应用程序来接受来自一组大约 10000 个客户端的 TCP 连接(实际上是具有 GPRS 接口的现场设备)。

接受连接的部分是这样的:

public async System.Threading.Tasks.Task AcceptConnections()
{
    _listenerProxy.StartListen();
    var taskList = new List<System.Threading.Tasks.Task>();
    var application = _configManager.Configuration.GetSection("application");
    var maxTasks = int.Parse(application["MaxTasksPerListener"]);
    while (true)
    {
        if (taskList.Count > maxTasks)
        {
            _logger.Info($"Waiting for handling task to be completed {_listenerProxy.Port}.");
            await System.Threading.Tasks.Task.WhenAny(taskList.ToArray()).ConfigureAwait(false);
            taskList.RemoveAll(t => t.IsCompleted); // at least one task affected, but maybe more
        }
        _logger.Info($"Waiting for client to be accepted on port {_listenerProxy.Port}.");
        (var readStream, var writeStream, var sourceIP, var sourcePort) = await _listenerProxy.AcceptClient().ConfigureAwait(false);
        _logger.Info($"Client accepted on port {_listenerProxy.Port}: IP:{sourceIP}, Port:{sourcePort}");

        var clientContext = new ClientContext<TSendMessage, TReceiveMessage>
        {
            SourceAddress = sourceIP,
            SourcePort = sourcePort,
            DataAdapter = DataAdapterFactory<TSendMessage, TReceiveMessage>.Create(readStream, writeStream)
        };
        taskList.Add(HandleClient(clientContext));
    }
}

HandleClient 方法定义为:

public async System.Threading.Tasks.Task HandleClient(ClientContext<TSendMessage, TReceiveMessage> clientContext);

我希望能够并行处理多达预定义数量的请求(处理函数是HandleClient)。客户端将连接,发送少量数据,然后关闭连接。由于整个项目都是异步的,所以我很想尝试在这部分使用异步方法。

我很确定不建议使用此解决方案,但我不知道如何以更好的方式完成。我在这里找到了一个非常相关的主题:如何在新线程上运行任务并立即返回给调用者?

Stephen Cleary 在那里发表了评论:

好吧,我建议的第一件事是尝试一个更简单的练习。说真的,异步 TCP 服务器是您可以选择的最复杂的应用程序之一。

那么,在这种情况下,“正确”的方法是什么?我知道,我的方法“有效”,但我有一种不好的感觉,尤其HandleClient是或多或少是“一劳永逸”。我对生成的任务感兴趣的唯一原因是“限制”吞吐量(最初它是异步无效的)。事实上,我的问题与链接中的 TO 完全相同。但是,我仍然不知道这种方法中最大的问题是什么。

我将不胜感激建设性的提示...谢谢。

标签: c#socketsasynchronousasync-awaittcplistener

解决方案


一句话:红隼。将所有这些问题转移到此,同时获得许多其他好处,例如高级缓冲区生命周期管理,以及面向异步的设计。这里的关键 API 是UseConnectionHandler<T>,例如:

public static IWebHostBuilder CreateHostBuilder(string[] args) =>
    WebHost.CreateDefaultBuilder(args).UseKestrel(options =>
    {   // listen on port 1000, using YourHandlerHere for the handler
        options.ListenLocalhost(1000, o => o.UseConnectionHandler<YourHandlerHere>());
    }).UseStartup<Startup>();

您可以在此处看到一个微不足道但可运行的示例(嘿,如果您将功能最多的类似 redis 的服务器称为“微不足道”),或者我在私人 github 存储库中有其他示例,我可能会发送给您,包括一个示例,其中所有您需要做的是实现帧解析器(库代码处理其他所有内容,并且适用于 TCP 和 UDP)。

至于“油门”——听起来像是一个异步信号量;SemaphoreSlimWaitAsync


推荐阅读