文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C#怎么使用CallContext缓存线程数据

2023-06-30 13:39

关注

本篇内容主要讲解“C#怎么使用CallContext缓存线程数据”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C#怎么使用CallContext缓存线程数据”吧!

一、CallContext 概述

命名空间:System.Runtime.Remoting.Messaging

CallContext 用于提供与执行代码路径一起传送的属性集,直白讲就是:提供线程(多线程/单线程)代码执行路径中数据传递的能力。

当对另一个 AppDomain 中的对象进行远程方法调用时,CallContext 类将生成一个与该远程调用一起传播的 LogicalCallContext 实例。只有公开 ILogicalThreadAffinative 接口并存储在 CallContext 中的对象被在 LogicalCallContext 中传播到 AppDomain 外部。

CallContext成员

GetData、SetData

LogicalSetData、LogicalGetData

二、 CallContext不跨线程传播的方法:GetData、SetData

可以利用CallContext 实现单例,默认情况下,CallContext 的数据不跨线程传播。

在处理多组件共用Context时非常有用,比如常见的EF 可以将实例的DBEntity存储在其中,可以一次访问只实例化一次,便于管理且不用多次实例访问对象

public static class DbContextHelper{    private static DbContext context = null;    private const string SessionKey_DbContext = "Entities";    public static DbContext GetDbContext()    {        if (CallContext.GetData(SessionKey_DbContext) == null)        {            CallContext.SetData(SessionKey_DbContext, new Entities());        }        return CallContext.GetData(SessionKey_DbContext) as Entities;    }}

类单例

void Main(){    MyAppContext.Current.FirstName = "a";    Console.Write(MyAppContext.Current.FirstName);}public class MyAppContext{    const string contextKey = "MyAppContext:ContextKey";    public string FirstName { get; set; }    public static MyAppContext Current    {        get        {            if (CallContext.GetData(contextKey) == null)            {                CallContext.SetData(contextKey, new MyAppContext());            }            return CallContext.GetData(SessionKey_DbContext) as MyAppContext;        }    }}

三、 CallContext跨线程传播的方法:ILogicalSetData、LogicalGetData

要让CallContext实现跨线程传播,可以调用CallContext的静态方法ILogicalSetData,或让上下文类实现ILogicalThreadAffinative 接口。

线程本地存储

线程池可能不会释放使用过的线程,导致多次执行之间可能共享数据(可以每次执行前重置线程本地存储的数据)。

for (var i = 0; i < 10; i++){    Thread.Sleep(10);    Task.Run(() =>    {        var slot = Thread.GetNamedDataSlot("test");        if (slot == null)        {            Thread.AllocateNamedDataSlot("test");        }        if (Thread.GetData(slot) == null)        {            Thread.SetData(slot, DateTime.Now.Millisecond);        }        Console.WriteLine(Thread.CurrentThread.ManagedThreadId + ":" + Thread.GetData(slot));    });}

结果

C#怎么使用CallContext缓存线程数据

调用上下文

每次执行的数据是完全隔离的,非常符合我们的期望。但是,如果我们期望调用期间又开启了一个子线程,如何让子线程访问父线程的数据呢?这就需要使用到:“逻辑调用上下文”。

Console.WriteLine("测试:CallContext.SetData");for (var i = 0; i < 10; i++){    Thread.Sleep(10);    Task.Run(() =>    {        if (CallContext.GetData("test") == null)        {            CallContext.SetData("test", DateTime.Now.Millisecond);        }        Console.WriteLine(Thread.CurrentThread.ManagedThreadId + ":" + CallContext.GetData("test"));    });}

结果

C#怎么使用CallContext缓存线程数据

每次执行的数据是完全隔离的,非常符合我们的期望。

逻辑调用上下文

如果我们期望调用期间又开启了一个子线程,如何让子线程访问父线程的数据呢?这就需要使用到:“逻辑调用上下文”。

注意 ExecutionContext.SuppressFlow(); 和ExecutionContext.RestoreFlow();它们分别能阻止传播和重置传播,默认是允许传播的。

Console.WriteLine("测试:CallContext.SetData");Task.Run(() =>{    CallContext.SetData("test", "段光伟");    Console.WriteLine(Thread.CurrentThread.ManagedThreadId + ":" + CallContext.GetData("test"));    Task.Run(() =>    {        Console.WriteLine(Thread.CurrentThread.ManagedThreadId + ":" + CallContext.GetData("test"));    });});Thread.Sleep(100);Console.WriteLine("测试:CallContext.LogicalSetData");Task.Run(() =>{    CallContext.LogicalSetData("test", "段光伟");    Console.WriteLine(Thread.CurrentThread.ManagedThreadId + ":" + CallContext.LogicalGetData("test"));    Task.Run(() =>    {        Console.WriteLine(Thread.CurrentThread.ManagedThreadId + ":" + CallContext.LogicalGetData("test"));    });    ExecutionContext.SuppressFlow();    Task.Run(() =>    {        Console.WriteLine("SuppressFlow 之后:" + CallContext.LogicalGetData("test"));    });    ExecutionContext.RestoreFlow();    Task.Run(() =>    {        Console.WriteLine("RestoreFlow 之后:" + CallContext.LogicalGetData("test"));    });});

输出

C#怎么使用CallContext缓存线程数据

四、Web中的CallContext

HttpContext.Current(包括Session)的存储是基于当前线程的CallContext,在非请求处理线程(即其他线程)是无法获取当前HttpContext的(不跨线程传播)。

到此,相信大家对“C#怎么使用CallContext缓存线程数据”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯