ThreadLocal详解【使用场景】
转:
么是ThreadLocal
根据JDK文档中的解释:ThreadLocal的作用是提供线程内的局部变量,这种变量在多线程环境下访问时能够保证各个线程里变量的独立性。
从这里可以看出,引入ThreadLocal的初衷是为了提供线程内的局部变量
ThreadLocal
不是一个线程,而是一个线程的本地化对象。当某个变量在使用 ThreadLocal
进行维护时,ThreadLocal
为使用该变量的每个线程分配了一个独立的变量副本。
每个线程可以自行操作自己对应的变量副本,而不会影响其他线程的变量副本。
API 方法
ThreadLocal
的 API 提供了如下的 4 个方法。
1)protected T initialValue()
返回当前线程的局部变量副本的变量初始值。
2)T get()
返回当前线程的局部变量副本的变量值,如果此变量副本不存在,则通过 initialValue()
方法创建此副本并返回初始值。
3)void set(T value)
设置当前线程的局部变量副本的变量值为指定值。
4)void remove()
删除当前线程的局部变量副本的变量值。
在实际使用中,我们一般都要重写 initialValue()
方法,设置一个特定的初始值。
关于initialValue的初始化。本人尝试了多种方式:
1
2
3
4
5
6
7
8
|
//new ThreadLocal方式:不推荐 final ThreadLocal<String> commandThreads = new ThreadLocal<String>() { @Override protected String initialValue() { return "execute :" +System.currentTimeMillis(); } }; System.out.println(commandThreads.get()); |
1
2
3
4
5
|
//withInitial方式: ThreadLocal<String> commandThreadnew = // ThreadLocal.withInitial(()-> "execute :"+System.currentTimeMillis()); ThreadLocal.withInitial(()-> new String( "execute :" +System.currentTimeMillis())); System.out.println(commandThreadnew.get()); |
1
2
3
4
5
6
7
8
9
10
|
//(new Supplier<String>(){}方式 推荐 ThreadLocal<String> commandThreadnew1 = ThreadLocal.withInitial( new Supplier<String>() { @Override public String get() { return "execute :" +System.currentTimeMillis(); } }); System.out.println( commandThreadnew1.get()); |
以下是关于ThreadLocal 解决多线程变量共享问题:
存在争议点:
ThreadLocal到底能不能解决共享对象的多线程访问问题?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
|
package com.wuxianjiezh.demo.threadpool; public class MainTest { public static void main(String[] args) { Bank bank = new Bank(); Thread xMThread = new Thread(() -> bank.deposit( 200 ), "小明" ); Thread xGThread = new Thread(() -> bank.deposit( 200 ), "小刚" ); Thread xHThread = new Thread(() -> bank.deposit( 200 ), "小红" ); xMThread.start(); xGThread.start(); xHThread.start(); } } class Bank { private int money = 1000 ; public void deposit( int money) { String threadName = Thread.currentThread().getName(); System.out.println(threadName + "--当前账户余额为:" + this .money); this .money += money; System.out.println(threadName + "--存入 " + money + " 后账户余额为:" + this .money); try { Thread.sleep( 1000 ); } catch (InterruptedException e) { e.printStackTrace(); } } } |
运行结果:存在多线程输出结果混乱
1
2
3
4
5
6
|
小明--当前账户余额为: 1000 小红--当前账户余额为: 1000 小红--存入 200 后账户余额为: 1400 小刚--当前账户余额为: 1000 小刚--存入 200 后账户余额为: 1600 小明--存入 200 后账户余额为: 1200 |
使用 ThreadLocal
保存对象的局部变量。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
public class MainTest { public static void main(String[] args) { Bank bank = new Bank(); Thread xMThread = new Thread(() -> bank.deposit( 200 ), "小明" ); Thread xGThread = new Thread(() -> bank.deposit( 200 ), "小刚" ); Thread xHThread = new Thread(() -> bank.deposit( 200 ), "小红" ); xMThread.start(); xGThread.start(); xHThread.start(); } } class Bank { // 初始化账户余额为 100 ThreadLocal<Integer> account = ThreadLocal.withInitial( new Supplier<Integer>() { @Override public Integer get() { return 1000 ; } }); public void deposit( int money) { String threadName = Thread.currentThread().getName(); System.out.println(threadName + "--当前账户余额为:" + account.get()); account.set(account.get() + money); System.out.println(threadName + "--存入 " + money + " 后账户余额为:" + account.get()); try { Thread.sleep( 1000 ); } catch (InterruptedException e) { e.printStackTrace(); } } } |
运行结果为:
1
2
3
4
5
6
7
|
小明--当前账户余额为: 1000 小红--当前账户余额为: 1000 小红--存入 200 后账户余额为: 1200 小刚--当前账户余额为: 1000 小刚--存入 200 后账户余额为: 1200 小明--存入 200 后账户余额为: 1200 可以看到,我们要的效果达到了。各线程间同时操作自己的变量,相互间没有影响。 |
ThreadLocal 与 Thread 同步机制的比较
同步机制采用了以时间换空间方式,通过对象锁保证在同一个时间,对于同一个实例对象,只有一个线程访问。
ThreadLocal
采用以空间换时间方式,为每一个线程都提供一份变量,各线程间同时访问互不影响。
转载请注明出处:https://segmentfault.com/a/1190000009236777
ThreadLocal详解【使用场景】的更多相关文章
- 深入解析ThreadLocal 详解、实现原理、使用场景方法以及内存泄漏防范 多线程中篇(十七)
简介 从名称看,ThreadLocal 也就是thread和local的组合,也就是一个thread有一个local的变量副本 ThreadLocal提供了线程的本地副本,也就是说每个线程将会拥有一个 ...
- android Handler机制之ThreadLocal详解
概述 我们在谈Handler机制的时候,其实也就是谈Handler.Message.Looper.MessageQueue之间的关系,对于其工作原理我们不做详解(Handler机制详解). Messa ...
- Java并发编程:线程封闭和ThreadLocal详解
转载请标明出处: http://blog.csdn.net/forezp/article/details/77620769 本文出自方志朋的博客 什么是线程封闭 当访问共享变量时,往往需要加锁来保证数 ...
- ThreadLocal详解(实现多线程同步访问变量)
ThreadLocal翻译成中文比较准确的叫法应该是:线程局部变量. 这个玩意有什么用处,或者说为什么要有这么一个东东?先解释一下,在并发编程的时候,成员变量如果不做任何处理其实是线程不安全的,各个线 ...
- ThreadLocal详解
ThreadLocal翻译成中文比较准确的叫法应该是:线程局部变量. 这个玩意有什么用处,或者说为什么要有这么一个东东?先解释一下,在并发编程的时候,成员变量如果不做任何处理其实是线程不安全的,各个线 ...
- 并发系列(2)之 ThreadLocal 详解
本文将主要结合源码讲述 ThreadLocal 的使用场景和内部结构,以及 ThreadLocalMap 的内部结构:另外在阅读文本之前只好先了解一下引用和 HashMap 的相关知识,可以参考 Re ...
- ThreadLocal详解,ThreadLocal源码分析,ThreadLocal图解
本文脉路: 概念阐释 ----> 原理图解 ------> 源码分析 ------> 思路整理 ----> 其他补充. 一.概念阐述. ThreadLocal 是一个为 ...
- 【Java深入研究】7、ThreadLocal详解
ThreadLocal翻译成中文比较准确的叫法应该是:线程局部变量. 这个玩意有什么用处,或者说为什么要有这么一个东东?先解释一下,在并发编程的时候,成员变量如果不做任何处理其实是线程不安全的,各个线 ...
- ThreadLocal详解,处理成员变量线程不安全的情况
ThreadLocal翻译成中文比较准确的叫法应该是:线程局部变量. 这个玩意有什么用处,或者说为什么要有这么一个东东?先解释一下,在并发编程的时候,成员变量如果不做任何处理其实是线程不安全的,各个线 ...
随机推荐
- C#编程 LINQ查询
LINQ查询表达式 约束 LINQ查询表达式必须以from子句开头,以select或group子句结束 关键字 from...in...:指定要查找的数据以及范围变量,多个from子句则表示从多个数据 ...
- 将服务端select设置为非阻塞,处理更多业务
服务端代码: #include<WinSock2.h> #include<Windows.h> #include<vector> #include<stdio ...
- ubuntu下log4cxx安装使用
需要安装log4cxx,安装的过程中可是充满了坎坷...最大的问题是在make log4cxx时,总是报undefined XML什么什么的错误,查了一下也没解决了,然后把apr-utils删了重新装 ...
- 01—mybatis开山篇
什么是 MyBatis ? MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.M ...
- java中list和map的底层实现原理
Collection(单列集合) List(有序,可重复) ArrayList 底层数据结构是数组,查询快,增删慢 线程不安全,效率高 Vector 底层数据结构是数组,查询快,增删慢 线程安全,效率 ...
- p4434 [COCI2017-2018#2] Usmjeri
思路 并查集的好题 考虑到求满足条件限制的方案数,显然观察样例可知结果就是2^x,x是互不影响的边的集合数量 然后考虑如何求互不影响的边的集合数量 可以使用并查集,用i和i+n表示这个点的父亲连向它的 ...
- onpageshow、onpagehide、onload、onunload
onpageshow :在用户浏览网页时触发, 在页面从浏览器缓存中读取时也触发 通过event.persisted 来判断, 如果页面从浏览器的缓存中读取该属性返回 ture,否则返回 false ...
- js原型补充
js定义函数: <script> function A() {} let a1 = new A(); let a2 = new A(); // 为A类添加原型 => 类似于类属性 A ...
- swoole 协程channel乱测
channel和数组差不多,可以被用作队列,属性capacity是设置容量,isEmpty() isFull() 用来判断队列是空还是满,push()加入队列 pop()弹出队列 interface ...
- 一个列表实现__iter__和__next__方法的例子
x = ['厉智','陈培昌','程劲','徐晓冬'].__iter__() #这非得这么写不可,否则无法调用下面的__next__()方法,切记! print(x.__next__()) print ...