NET 4.6引入了AsyncLocal<T>类,用于沿着异步控制流流动环境数据.我之前已经使用CallContext.LogicalGet/SetData实现了这个目的,我想知道这两者在语义上是否以及在哪些方面是不同的(除了明显的API差异,比如强类型和不依赖字符串键).

推荐答案

语义基本相同.两者都存储在ExecutionContext中,并通过异步调用流动.

区别在于API更改(正如您所描述的)以及为值更改注册回调的能力.

从技术上讲,实现上有很大的不同,因为每次复制CallContext时(使用CallContext.Clone)都会克隆CallContext,而AsyncLocal的数据保存在ExecutionContext._localValues字典中,只复制引用,不需要任何额外的工作.

为了确保当您更改AsyncLocal的值时,更新只影响当前流,将创建一个新字典,并将所有现有值浅复制到新的字典中.

这种差异对性能可能是好的,也可能是坏的,这取决于AsyncLocal的使用位置.

现在,正如Hans Passant在 comments 中提到的,CallContext最初用于远程处理,在不支持远程处理的地方(例如Net Core)不可用,这可能就是为什么将AsyncLocal添加到框架中:

#if FEATURE_REMOTING
    public LogicalCallContext.Reader LogicalCallContext 
    {
        [SecurityCritical]
        get { return new LogicalCallContext.Reader(IsNull ? null : m_ec.LogicalCallContext); } 
    }

    public IllogicalCallContext.Reader IllogicalCallContext 
    {
        [SecurityCritical]
        get { return new IllogicalCallContext.Reader(IsNull ? null : m_ec.IllogicalCallContext); } 
    }
#endif

注意:VisualStudioSDK中还有一个AsyncLocal,基本上是CallContext的包装,它显示了这些概念的相似性:Microsoft.VisualStudio.Threading.

.net相关问答推荐

SLN配置文件:映射问题

如何从 tshark 的 stderr 捕获实时数据包计数?

等待时 Blazor 服务器按钮刷新

在本地运行 Azure 函数会在 .NET7 升级后出现无运行时错误

线程安全性的单元测试?

Erlang 的让它崩溃的哲学 - 适用于其他地方吗?

如何判断 IOException 是否为 Not-Enough-Disk-Space-Exception 类型?

托管和非托管代码、内存和大小有什么区别?

基于多个字符分隔符拆分字符串

什么是 Hashtable 的通用版本?

是否有 TLS 1.2 的 .NET 实现?

如何异步 Files.ReadAllLines 并等待结果?

设置 System.Drawing.Color 值

C# - 你如何停止计时器?

何时何地使用 GetType() 或 typeof()?

无锁多线程适用于真正的线程专家

.NET 中的 java.lang.IllegalStateException?

如何为 Dapper 查询动态创建参数

从 bcp 客户端接收到 colid 6 的无效列长度

判断数据表中是否包含空值的最佳方法