InheritableThreadLocal的使用问题

单例对象,在多线程环境中使用时,为了避免线程冲突,大家都知道要通过ThreadLocal来管理该对象,为每个线程保存该对象副本,spring中管理bean、hibernated的SessionFactory管理Session都是这么做的,但是ThreadLocal有一个子类InheritableThreadLocal,使得父子线程之间能够共享实例,如
private static ThreadLocal<Context> configHolder = new InheritableThreadLocal(); 

在web应用场景中能够解决一些问题,但是目前很多应用都会使用quartz,比如在晚上让系统来处理一些任务,但是当这些任务同时执行时,由于有共同的quartz的父任务,导致这些任务来操作通过InheritableThreadLocal管理的对象时,可能导致线程冲突,所以在这种场景下要慎用InheritableThreadLocal。
当时本人碰到的一个场景是,晚上系统来统计系统的访问统计信息,由于本系统是一个应用管理多个网站,每个网站的信息是通过一个Context来保存访问具体哪个网站的,当执行quartz后,有些网站的关键词显然不是这个网站的,当时觉得问题很诡异,最后通过加了大量输出时发现,在一个网站的统计还没结束时,Context里的网站已经被切换了,显然这不是自己的任务切换的,最终发现同时有另外的Quartz任务也在操作Context对象,这类问题很难定位原因,当时搞了很久才发现这个原因

猜你喜欢

转载自pocketduck.iteye.com/blog/1060822