一个实现单线程/多线程下代码调用链中传递数据的处理类: CallContext

-- 多线程下函数调用链中传递数据的处理类: CallContext
【官网】:https://docs.microsoft.com/zh-cn/dotnet/api/system.runtime.remoting.messaging.callcontext?redirectedfrom=MSDN&view=netframework-

应用场景

当需要提供与执行代码路径一起传送的属性集(通俗说: 需要在代码调用链环境下传递数据)时可以使用CallContext。

基础资源

.net framework下依赖System.Runtime.Remoting.Messaging.CallContext  ,.net core下需自行实现(本文已包含相关实现代码)

使用须知

CallContext 专用集合对象,它类似于方法调用的线程本地存储,并提供对每个逻辑执行线程唯一的数据槽。 不会在其他逻辑线程上跨调用上下文共享槽。 CallContext当对象向下移动并备份执行代码路径,并沿着路径按各个对象检查时,可以将对象添加到中。

配置步骤

关于System.Runtime.Remoting.Messaging.CallContext的注意事项

1、GetData、SetData只能用于单线程环境,如果发生了线程切换,存储的数据也会随之丢失

2、可以用于同一线程中的不同地方,传递数据。

3.  FreeNamedDataSlot只能清除当前线程的数据槽,不能清除子线程的数据槽;

4、LogicalSetData、LogicalGetData可用于在多线程环境下传递数据;

5、FreeNamedDataSlot清除当前线程,之前已经运行子任务,不受影响.

6.   对于LogicalSetData, 只能在调用链上自上而下设置会生效,反向不生效。




常见问题

快速入门

A).net framework下CallContext的应用。

a1)对System.Runtime.Remoting.Messaging.CallContext的调用及线程内数据槽的释放。


using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Remoting.Messaging;
using System.Text;
using System.Threading;
using System.Threading.Tasks;

namespace DonetFrm.Test.Demo
{
    /// <summary>
    /// author: http://config.net.cn
    /// project site:https://github.com/configlab/ResearchNotes-Demo
    /// description:use CallContext in  .net framework with System.Runtime.Remoting.Messaging.CallContext
    /// create date: 2021-10-6
    /// </summary>
    public class CallContextTest_Framework
    {
        public void Test()
        {
            CallContext.LogicalSetData("entry_method", "test");
            Console.WriteLine($"debug-main-1:threadId={Thread.CurrentThread.ManagedThreadId},entry_method={CallContext.LogicalGetData("entry_method")}");
            Task _task = new Task(() => {
                CallContext.LogicalSetData("set_from_task", "true");
                Console.WriteLine($"debug-task-1:threadId={Thread.CurrentThread.ManagedThreadId},entry_method={CallContext.LogicalGetData("entry_method")}");
                CallContext.FreeNamedDataSlot("entry_method");
                Console.WriteLine($"debug-task-2:threadId={Thread.CurrentThread.ManagedThreadId},entry_method={CallContext.LogicalGetData("entry_method")}");
            });
            _task.Start();
            new Thread(new ThreadStart(() => {
                CallContext.LogicalSetData("set_from_subthread", "true");
                Console.WriteLine($"debug-subthread-1:threadId={Thread.CurrentThread.ManagedThreadId},entry_method={CallContext.LogicalGetData("entry_method")}");
                CallContext.FreeNamedDataSlot("entry_method");
                Console.WriteLine($"debug-subthread-2:threadId={Thread.CurrentThread.ManagedThreadId},entry_method={CallContext.LogicalGetData("entry_method")}");
            })).Start();
            Thread.Sleep(5000);
            Console.WriteLine($"debug-main-2:threadId={Thread.CurrentThread.ManagedThreadId},entry_method={CallContext.LogicalGetData("entry_method")},set_from_task={CallContext.GetData("set_from_task")},set_from_subthread={CallContext.GetData("set_from_subthread")}");
        }
    }
}


【效果如图】

callcontext

B).net core下CallContext的应用。

b1)自定义实现的CallContext 源码。


using System;
using System.Collections.Concurrent;
using System.Threading;

namespace ConfigLab.CallContext
{

    /// author: http://config.net.cn
    /// project site:https://github.com/configlab/ResearchNotes-Demo
    /// description:Provides a way to set contextual data that flows with the call and async context of a test or invocation.
    /// create date: 2021-10-6
    /// refer: https://www.cazzulino.com/callcontext-netstandard-netcore.html
    public static class CallContext
    {
        static ConcurrentDictionary<string, AsyncLocal<object>> state = new ConcurrentDictionary<string, AsyncLocal<object>>();

        /// <summary>
        /// Stores a given object and associates it with the specified name.
        /// </summary>
        /// <param name="name">The name with which to associate the new item in the call context.</param>
        /// <param name="data">The object to store in the call context.</param>
        public static void SetData(string name, object data) =>
            state.GetOrAdd(name, _ => new AsyncLocal<object>()).Value = data;

        /// <summary>
        /// Retrieves an object with the specified name from the <see cref="CallContext"/>.
        /// </summary>
        /// <param name="name">The name of the item in the call context.</param>
        /// <returns>The object in the call context associated with the specified name, or <see langword="null"/> if not found.</returns>
        public static object GetData(string name) =>
            state.TryGetValue(name, out AsyncLocal<object> data) ? data.Value : null;
    }
}	

b2)基于自定义实现的CallContext的应用。

callcontext


[附:本文源码地址]

https://github.com/configlab/ResearchNotes-Demo

参考资料