如何在并发字典中存储频道

时间:2021-02-11 22:50:21

标签: c# system.threading.channels

这里有一个相当棘手的问题,但我试图理解的概念相当简单。

我有一个公共静态并发字典,可以从其他类和线程访问:

public static readonly ConcurrentDictionary<int, object> tcpServerChannelDictionary = new ConcurrentDictionary<int, object>();

我有一个特定的后台线程,我最初在其中创建了一个新的 C# 通道并将实例存储在字典中。在同一个线程中(在同一个方法中),我开始异步等待消息。从方法中删除了一些不必要的代码膨胀:

var myChannel = Channel.CreateUnbounded<string>();
            tcpServerChannelDictionary.TryAdd(int, myChannel);

await foreach (var data in myChannel.Reader.ReadAllAsync())
{
    // Do something with message
}

第一个绊脚石是我无法使用“对象”定义静态字典作为通道,即 ConcurrentDictionary...作为类型参数”所以这就是为什么我改为声明一个对象类型。

这似乎在没有 VS 代码警告的情况下工作,直到我尝试从另一个类(以及另一个线程)访问字典对象:

if (TcpServer.tcpServerChannelDictionary.TryGetValue(int, out Channel myChannel))
{
   await myChannel.Writer.WriteAsync(data);
}

enter image description here

我不知何故需要将对象投射到我试图访问的频道,以便我可以向它写一条消息,或者其他一些防止警告的解决方案。我是频道的新手,但立即看到使用它们会有多么强大。

1 个答案:

答案 0 :(得分:2)

Channel 类是一个带有工厂方法的辅助静态类。所有这些方法都返回一个 Channel<T>。在您的情况下,您正在创建 Channel<string> 对象,这是一种可以构造 Dictionary 的类型。

public static readonly ConcurrentDictionary<int, Channel<string>> tcpServerChannelDictionary = new();
//... 
if (TcpServer.tcpServerChannelDictionary.TryGetValue(int, out var /*Channel<string>*/ myChannel)) 
{
   await myChannel.Writer.WriteAsync(data);
}

现在,如果无论出于何种原因,您仍然需要使用 object 类型作为字典的值(也许您对 Channel<T> 有不同的通用参数,并且希望它们都在同一个集合中)您的out 变量仍然需要是 object 然后你需要转换它:

if (TcpServer.tcpServerChannelDictionary.TryGetValue(int, out var /*object*/ oChannel) && oChannel is Channel<string> myChannel)
{
   await myChannel.Writer.WriteAsync(data);
}