• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

java  中ThreadLocal实例分析

java 搞代码 4年前 (2022-01-05) 38次浏览 已收录 0个评论

这篇文章主要介绍了java 中ThreadLocal实例分析的相关资料,需要的朋友可以参考下

java  中ThreadLocal实例分析

从概念上理解,threadlocal使变量在多个线程中相互隔离实现线程安全,threadlocal包装的变量最终都专属于对应的每个线程,线程之间相互独立,用一个具体实现来说明:

 public interface Consumer { int consume(); } public class ComsumeThread implements Runnable { private Consumer consumer; public ComsumeThread(Consumer consumer) { this.consumer = consumer; } @Override public void run() { for(int i=0;i<10;i++){ System.out.println(Thread.currentThread().getName()+" After Consume left:"+consumer.consume()); } } } public class ConsumeClientA implements Consumer { private static int leftNum = 30; @Override public int consume() { int orgLeftNum = leftNum; Random random = new Random(System.currentTimeMillis()); try { Thread.sleep(random.nextInt(3)); } catch (InterruptedException e) { e.printStackTrace(); } orgLeftNum = orgLeftNum -1; leftNum = orgLeftNum; return leftNum; } public static void main(String[] args){ Consumer consumer = new ConsumeClientA(); Thread thread1 = new Thread(new ComsumeThread(consumer)); Thread thread2 = new Thread(new ComsumeThread(consumer)); Thread thread3 = new Thread(new ComsumeThread(consumer)); thread1.start(); thread2.start(); thread3.start(); } } 

ConsumeClientA是在没有做任何线程安全处理,结果如下:

 Thread-2 After Consume left:29 Thread-1 After Consume left:29 Thread-3 After Consume left:29 Thread-2 After Consume left:28 Thread-1 After Consume left:28 Thread-3 After Consume left:28 Thread-2 After Consume left:27 Thread-1 After Consume left:27 Thread-2 After Consume left:26 Thread-3 After Consume left:27 Thread-1 After Consume left:25 Thread-2 After Consume left:25 Thread-3 After Consume left:25 Thread-1 After Consume left:24 Thread-2 After Consume left:24 Thread-3 After Consume left:24 Thread-1 After Consume<em style="color:transparent">来源[email protected]搞@^&代*@码网</em> left:23 Thread-2 After Consume left:23 Thread-3 After Consume left:23 Thread-1 After Consume left:22 Thread-2 After Consume left:22 Thread-3 After Consume left:22 Thread-1 After Consume left:21 Thread-2 After Consume left:21 Thread-3 After Consume left:21 Thread-1 After Consume left:20 Thread-2 After Consume left:20 Thread-3 After Consume left:20 Thread-1 After Consume left:19 Thread-3 After Consume left:18 

增加threadlocal处理,每个线程相互独立,实现如下:

 public class ConsumeClientB implements Consumer { private ThreadLocal leftNumThreadLocal = new ThreadLocal(){ @Override protected Integer initialValue() { return 30; } }; @Override public int consume() { int orgLeftNum = leftNumThreadLocal.get(); Random random = new Random(System.currentTimeMillis()); try { Thread.sleep(random.nextInt(3)); } catch (InterruptedException e) { e.printStackTrace(); } orgLeftNum = orgLeftNum -1; leftNumThreadLocal.set(orgLeftNum); return leftNumThreadLocal.get(); } public static void main(String[] args){ Consumer consumer = new ConsumeClientB(); Thread thread1 = new Thread(new ComsumeThread(consumer)); Thread thread2 = new Thread(new ComsumeThread(consumer)); Thread thread3 = new Thread(new ComsumeThread(consumer)); thread1.start(); thread2.start(); thread3.start(); } } 

运行的结果如下:

 Thread-1 After Consume left:29 Thread-3 After Consume left:29 Thread-2 After Consume left:29 Thread-1 After Consume left:28 Thread-3 After Consume left:28 Thread-2 After Consume left:28 Thread-1 After Consume left:27 Thread-3 After Consume left:27 Thread-2 After Consume left:27 Thread-1 After Consume left:26 Thread-3 After Consume left:26 Thread-2 After Consume left:26 Thread-1 After Consume left:25 Thread-3 After Consume left:25 Thread-2 After Consume left:25 Thread-1 After Consume left:24 Thread-3 After Consume left:24 Thread-2 After Consume left:24 Thread-1 After Consume left:23 Thread-3 After Consume left:23 Thread-2 After Consume left:23 Thread-1 After Consume left:22 Thread-3 After Consume left:22 Thread-2 After Consume left:22 Thread-1 After Consume left:21 Thread-3 After Consume left:21 Thread-2 After Consume left:21 Thread-1 After Consume left:20 Thread-3 After Consume left:20 Thread-2 After Consume left:20 

每个线程拥有自己的独立变量,相互隔离实现线程安全。

那ThreadLocal是怎样实现这种线程隔离的线程安全的呢?

从ThreadLocal源码可以看到,真正实现线程隔离,与线程挂钩的,其实是ThreadLocal.ThreadLocalMap这个实现类,最明显的体现就在于Thread类源码的这样一个变量申明说明了ThreadLocal.ThreadLocalMap与Thread的关系:

 ThreadLocal.ThreadLocalMap threadLocals, inheritableThreadLocals;

Thread类是包含threadLocals对象的,ThreadLocal的具体实现就是根据提供的get,set等接口,对当前thread的threadLocals变量进行相关操作的,如get操作代码如下:

 public T get() { Thread t = Thread.currentThread(); ThreadLocalMap map = getMap(t); if (map != null) { ThreadLocalMap.Entry e = map.getEntry(this); if (e != null) return (T)e.value; } return setInitialValue(); } ThreadLocal.ThreadLocalMap getMap(Thread t) { return t.threadLocals; } 

可以看到,getMap()方法就是从当前thread获取对应的threadLocals变量,然后从这个ThreadLocal.ThreadLocalMap类型的threadLocals变量中获取对应线程中该ThreadLocal对象对应的变量值。

set方法的操作也是一样:

 public void set(T value) { Thread t = Thread.currentThread(); ThreadLocal.ThreadLocalMap map = getMap(t); if(map != null) { map.set(this, value); } else { this.createMap(t, value); } } void createMap(Thread t, T firstValue) { t.threadLocals = new ThreadLocalMap(this, firstValue); } 
 static class Entry extends WeakReference { Object value; Entry(ThreadLocal var1, Object var2) { super(var1); this.value = var2; } } 

ThreadLocalMap中存的是内部类Entry的数组,Entry是继承WeakReference实现,WeakReference的好处是保存对象引用,而又不干扰该对象被GC回收,线程执行完回收threadLocals变量时不会受到Entry封装的变量的干扰。

而且ThreadLocalMap中的key是ThreadLocal,所以一个ThreadLocal对象只能在一个Thread对象中保存一个ThreadLocal的value。

综上,很多人说ThreadLocal的实现是ThreadLocalMap中存Thread对象为key,变量为value的map结构,其实是错误的。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

以上就是java  中ThreadLocal实例分析的详细内容,更多请关注gaodaima搞代码网其它相关文章!


搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:java  中ThreadLocal实例分析

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址