使用异步任务不流动的逻辑CallContext值

Mic*_* M. 6 c# asynchronous task-parallel-library

根据我读过的所有内容,以下测试方法应该通过.我试图理解它失败的原因.私有异步方法中的第一个断言按预期传递.但是,一旦任务返回并等待.检索时,CallContext中设置的值现在为null.

    [TestMethod]
    public void LogicalCallContextBlockingTest()
    {
        PerformSimpleAsyncWork().Wait();

        var result = CallContext.LogicalGetData("test");

        Assert.AreEqual(result, "expected");
    }       

    private async Task PerformSimpleAsyncWork()
    {
        await Task.Run(() =>
            {
                System.Threading.Thread.Sleep(100);
                CallContext.LogicalSetData("test", "expected");

                var result = CallContext.LogicalGetData("test");

                Assert.AreEqual(result, "expected");

            });   
    }
Run Code Online (Sandbox Code Playgroud)

Bra*_*don 11

使用async关键字修饰的方法在调用时创建子上下文.对此子上下文所做的任何更改都不会传播到父上下文.

因此,PerformSimpleAsyncWork获取一个子上下文,该子上下文可以看到调用者放入上下文中的任何内容,但是它所做的任何更改都不可用于调用者(LogicalCallContextBlockingTest).

如果您想了解更多信息,Stephen Cleary 对此行为有一个很好的写作.