谈谈ThreadStatic

可能经常做多线程、线程池的童鞋早就知道这种问题,原谅我一直对线程研究不深。

这个东西好像出现有一段时间了,不过最近我才用到,做的API的服务,用来保存当前请求的上下文内容,原来用过Thread.SetData,不过原来的用户量没这么大,没发现问题。

查了一些关于ThreadStatic的说明,有一些人说好,也有人说坑的,的确有坑,下面看个例子:

    class Program
    {
        [ThreadStatic]
        private static int? NowI;
        //private static string resultString = "";

        static void Main(string[] args)
        {
            //resultString = "";
            for (var i = 0; i < 100; i++)
            {
                int i1 = i;
                var t = new Task(() =>
                {
                    if(NowI == null) NowI = i1;
                    //var nowI = Thread.GetData(Thread.GetNamedDataSlot("NowI"));
                    //if (nowI == null)
                    //{
                    //    nowI = i1;
                    //    Thread.SetData(Thread.GetNamedDataSlot("NowI"), i1);
                    //}

                    Console.WriteLine(string.Format("第{0}次循环, i值:{1},线程ID:{2}\r\n", i1, NowI, Thread.CurrentThread.ManagedThreadId));
                });
                t.Start();
            }
            //Console.WriteLine(resultString.ToString());
            Console.Read();
        }
    }

  NowI保存在ThreadStatic中,循环100次,开100个线程,运行结果(结果没排序,不过已经可以看出问题了):

第1次循环, i值:1,线程ID:12

第2次循环, i值:1,线程ID:12

第3次循环, i值:3,线程ID:15

第0次循环, i值:0,线程ID:6

第5次循环, i值:5,线程ID:17

第4次循环, i值:4,线程ID:14

第6次循环, i值:6,线程ID:11

第12次循环, i值:6,线程ID:11

第13次循环, i值:6,线程ID:11

第14次循环, i值:6,线程ID:11

第15次循环, i值:6,线程ID:11

第16次循环, i值:6,线程ID:11

第17次循环, i值:6,线程ID:11

第18次循环, i值:6,线程ID:11

第19次循环, i值:6,线程ID:11

第20次循环, i值:6,线程ID:11

第21次循环, i值:6,线程ID:11

第22次循环, i值:6,线程ID:11

第23次循环, i值:6,线程ID:11

第24次循环, i值:6,线程ID:11

......

可以看出, 线程ID一样的,保存的NowI变量的值是一样的,再猜,可能是GC没有回收垃圾,然后在new Task最后加上GC.Collect,然而并没有什卵用。

问题原因是这样的:当线程重用时,垃圾回收并没有回收上次的空间,其中变量值依然存在,如果不重新赋值,必然出现延用上次值的现象。

所以,解决办法就是在线程结束前把ThreadStatic变量清空。

再看结果:

第0次循环, i值:0,线程ID:10

第3次循环, i值:3,线程ID:12

第4次循环, i值:4,线程ID:10

第1次循环, i值:1,线程ID:11

第5次循环, i值:5,线程ID:12

第6次循环, i值:6,线程ID:10

第9次循环, i值:9,线程ID:10

第8次循环, i值:8,线程ID:12

第13次循环, i值:13,线程ID:12

......

这次就正常了。

不过,有没有办法可以统一对线程结束后进行处理呢,把所有这种变更重置一下,目前还没找到,希望大神们弄过的给个意见。

时间: 2024-08-03 03:05:02

谈谈ThreadStatic的相关文章

谈谈对CAP定理的理解

谈谈对CAP定理的理解 CAP定理的常规解释是任何分布式系统只能在一致性(Consitency),可用性(Availability)和分区容忍性(Partition Tolerance)中三选二.这个解释很让人费解,笔者在看了一些文章后谈谈我对它的理解,还请斧正. 从问题出发 假设我们用一台服务器A对外提供存储服务,为了避免这台服务器宕机导致服务不可用,我们又在另外一台服务器B上运行了同样的存储服务.每次用户在往服务器A写入数据的时候,A都往服务器B上写一份,然后再返回客户端.一切都运行得很好,

谈谈我眼中的安徽企腾信息技术集成有限公司

在这里工作一段时间后,说说自己眼中的安徽企腾,我尽量本着客观的原则,不带任何主观因素与感情色彩. 首先,先说说规模,所谓的分成售前与售后,其实就是虚张声势,售前不过是一些笔记本的售后维修,售后与售前基本上没什么关系.而售后,销售加上技术,最多时也不过7个人.平时一些项目,大一点,一些不能做的直接外包出去,小一点的项目,所谓的"施工队",就是全体总动员. 公司里做了好几块业务,其中一块是飕飕商用路由,刚开始,想的是多么好,但是最后却被浇了一盆冷水,市场根本不接受,唯一做了一家宾馆,还是连

谈谈JS里的{ }大括号和[ ]中括号的用法

谈谈JS里的{ }大括号和[ ]中括号的用法,理解后就可以看懂JSON结构了. 一.{ } 大括号,表示定义一个对象,大部分情况下要有成对的属性和值,或是函数. 如:var LangShen = {"Name":"Langshen","AGE":"28"}; 上面声明了一个名为"LangShen"的对象,多个属性或函数用,(逗号)隔开,因为是对象的属性, 所以访问时,应该用.(点)来层层访问:LangShe

《谈谈认同》这个文章有点晦涩 写的不太好

认同是一个人主观能动性的一个影响因素.至于人的主观能动性还有什么影响因素,先不讨论,而且我也没学过也没想过.在可选择事物的主观能动性上,认同应该算最基本也是影响最大的了.回到主题,我们来谈谈认同.        认同有三个表现:第一.口头认同.第二.行动认同.第三.心理认同.简单的理解,就是口头认同只说认同,但不做,事前一套事后一套.行动认同就是说也说了,干也干了,但心里不这样想.心理认同就是口手心合一.但我们都清楚认同其实从本质上讲只有一种,就是心理认同.只有心理认同才是真正的认同.但是在工作

谈谈vector容器的三种遍历方法

说明:本文仅供学习交流,转载请标明出处,欢迎转载! vector容器是最简单的顺序容器,其使用方法类似于数组,实际上vector的底层实现就是采用动态数组.在编写程序的过程中,常常会变量容器中的元素,那么如何遍历这些元素呢?本文给出三种遍历方法. 方法一:采用下标遍历 由于vector容器就是对一个动态数组的包装,所以在vector容器的内部,重载了[]运算符,函数原型为:reference operator [] (size_type n);所以我们可以采用类似于数组的方式来访问vector容

谈谈怎么使用JSONP

JSONP是什么,其实它是目前主流的实现跨域通信的解决方案. 因为我们都知道,在我们使用ajax去发起HTTP请求的时候,是不可以跨域的.也就是不能在不同域名下.端口等去请求.如果我们强行这么做的话,它是会报错. 这里举个例子. function createXHR() { var xhr = new XMLHttpRequest(); return xhr; } function ajax(url) { var xhr = createXHR(); xhr.onreadystatechange

你读到了什么:谈谈阅读的空与实

開始写点文字后,有些文章有人留言评论说:「都是些空话,正确的废话」,而有些文章有人又会留言评论说:「就喜欢这种.满满的干货.实在」. 究竟什么是空还是实蛮让我困惑的,思考了一段时间后今天就想谈谈我从阅读文章中究竟读到了些什么? 不管是别人评价是空话还是实在,于写作者我而言仅仅负责将自己思考后的观点清晰的表达,而至于是用「干货」还是「湿货」的形式来表达事实上取决于支撑观点的内容本身. 代入去想我知道评论说文章太空的人本身对阅读的期待是能从中收获实用甚或立马用的上的东西.凡是怀有相似期待的读者多半是

从面试题谈谈js的闭包,原型

最近群里有小伙伴分享了两道面试题,这里我谈谈自己的理解,废话不多说,上第一题: var n = 10; var obj = { n:20, fn:(function(){ this.n += 2; n *= 3; return function(){ this.n *= 2; n += 1; console.log(n) } })(n) } var fn = obj.fn; fn(); obj.fn() console.log(n,obj.n) 这个题目中,定义的obj对象的fn属性是个自执行的

谈谈我对Linux系统学习的历程回顾

众所周知,Windows 和Linux 是目前最流行的2个操作系统.Windows系统适合普通用户,它的优势是图形化界面,简单易用,使用起来门槛很低,很容易上手,所以,windows占有了大多数普通用户群体.而Linux 被誉为黑客的操作系统,因其稳定和命令行操作的高效性而广泛用于开发工作,占有绝大多数开发者群体.当然,关于这两大系统的优缺点,这里就不再赘述,我主要想谈谈我的Linux的学习历程参考书籍<Linux就该这么学>和心得体会. Linux初体验 一年前,我还不知道Linux为何物,