前言:工作中将要使用ThreadLocal,先学习总结一波。有不对的地方欢迎评论指出。

定义

  ThreadLocal并不是一个Thread,而是Thread的局部变量。这些变量不同于它们的普通对应物,因为访问某个变量(通过其 get 或 set 方法)的每个线程都有自己的局部变量,它独立于变量的初始化副本。

作用

  实现每一个线程都有自己的共享变量。

使用方法

  

  initialValue:返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的,默认就是null。

  remove方法:将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 1.5 新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。

源码解析

  ThreadLocal 构造方法:

  

  initialValue() 方法:

  

  get() 方法:

  

  第一行为获取线程的当前活动线程

  

  然后获取到当前线程的ThreadLocalMap 对象,再通过当前threadLocal来获取这个map对象的键值对,从而取出当前threadLocal中存的变量副本。

  如果ThreadLocalMap 对象为空,或这个map里面还没有存当前threadLocal的变量副本,则调用setInitialValue(); 

  set() 方法:

  

  如果当前线程里面有线程变量map,则给当前线程变量(this)设置值(value);如果没有,则创建当前线程的线程变量map,并设置值。

  getMap() 方法:

  

  

  getMap() 方法中,返回了当前线程的变量,threadLocals,类型为ThreadLocalMap。

  setInitialValue() 方法:

  

  setInitialValue() 方法,主要是设置初始化的 当前线程变量的变量副本。如果当前线程里面还没有 当前线程变量Map(ThreadLocalMap),则,初始化当前线程(thread)的线程变量Map

  createMap() 方法:

  

  初始化当前线程(thread)的线程变量Map

  ThreadLocalMap 内部类:

  

  构造方法:

  

  从这里可以看出,threadLocalMap里面存的key值就是 ThreadLocal 对象。

  remove() 方法:

  

举例 验证线程变量的隔离性

 /**
* 本地线程变量 test
* Created by yule on 2018/6/26 22:35.
*/
public class ThreadLocalTest {
public static void main(String[] args) throws InterruptedException {
ThreadDemo1 threadDemo1 = new ThreadDemo1();
threadDemo1.start(); Thread.sleep(100); ThreadDemo2 threadDemo2 = new ThreadDemo2();
threadDemo2.start(); ThreadLocalTools.stringThreadLocal.set("main设置值");
System.out.println(ThreadLocalTools.stringThreadLocal.get());
}
} class ThreadLocalTools{
public static ThreadLocal<String> stringThreadLocal = new ThreadLocal<>();
} class ThreadDemo1 extends Thread{
@Override
public void run() {
super.run();
for(int i = 0; i < 10; i++){
System.out.println(ThreadLocalTools.stringThreadLocal.get());
ThreadLocalTools.stringThreadLocal.set("ThreadDemo1设置值");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
} class ThreadDemo2 extends Thread{
@Override
public void run() {
super.run();
for(int i = 0; i < 10; i++){
System.out.println(ThreadLocalTools.stringThreadLocal.get());
ThreadLocalTools.stringThreadLocal.set("ThreadDemo2设置值");
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}

  输出的第一个为null是因为在set()方法前调用get()方法,会给出initialValue()方法的值,默认为null。

总结

  ThreadLocal是解决线程安全问题一个很好的思路,它通过为每个线程提供一个独立的变量副本解决了变量并发访问的冲突问题。在很多情况下,ThreadLocal比直接使用synchronized同步机制解决线程安全问题更简单,更方便,且结果程序拥有更高的并发性。

  ThreadLocal通常是类中的 private static 字段,它们希望将状态与某一个线程(例如,用户 ID 或事务 ID)相关联。在线程消失之后,其线程局部实例的所有副本都会被垃圾回收(除非存在对这些副本的其他引用)。

  ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal类中定义了一个ThreadLocalMap,每一个Thread中都有一个该类型的变量——threadLocals——用于存储每一个线程的变量副本,Map中元素的键为线程对象,而值对应线程的变量副本。

  通常我们通过匿名内部类的方式定义ThreadLocal的子类,提供初始的变量值。

Java 类 ThreadLocal 本地线程变量的更多相关文章

  1. Java Concurrency - ThreadLocal, 本地线程变量

    共享数据是多线程应用最常见的问题之一,但有时我们需要为每个线程保存一份独立的变量.Java API 提供了 ThreadLocal 来解决这个问题. 一个 ThreadLocal 作用的例子: imp ...

  2. ThreadLocal本地线程变量的理解

     一般的Web应用划分为展现层.服务层和持久层三个层次,在不同的层中编写对应的逻辑,下层通过接口向上层开放功能调用.在一般情况下,从接收请求到返回响应所经过的所有程序调用都同属于一个线程.       ...

  3. 假如java类里的成员变量是自身的对象

    假如java类里的成员变量是自身的对象,则新建该类对象时内存中怎么分配空间,我感觉似乎死循环了. 不过我想的肯定是错的,因为很多类的成员变量是自身对象,并且绝对无错,举个例子: Class A{ pr ...

  4. java类里的成员变量是自身的对象问题

    今晚看单例模式饿汉时想到一个问题:假如java类里的成员变量是自身的对象,则新建该类对象时内存中怎么分配空间,我感觉似乎死循环了.于是上网搜索了下,哈哈,果然有人早就思考过这个问题了,站在巨人的肩膀上 ...

  5. ThreadLocal = 本地线程?

    一.定义 ThreadLocal是JDK包提供的,从名字来看,ThreadLocal意思就是本地线程的意思. 1.1 是什么? 要想知道他是个啥,我们看看ThreadLocal的源码(基于JDK 1. ...

  6. 测试 Java 类的非公有成员变量和方法

    引言 对于软件开发人员来说,单元测试是一项必不可少的工作.它既可以验证程序的有效性,又可以在程序出现 BUG 的时候,帮助开发人员快速的定位问题所在.但是,在写单元测试的过程中,开发人员经常要访问类的 ...

  7. Flask中的ThreadLocal本地线程,上下文管理

    先说一下和flask没有关系的: 我们都知道线程是由进程创建出来的,CPU实际执行的也是线程,那么线程其实是没有自己独有的内存空间的,所有的线程共享进程的资源和空间,共享就会有冲突,对于多线程对同一块 ...

  8. java类中,成员变量赋值第一个进行,其次是静态构造函数,再次是构造函数

    如题是结论,如果有人问你Java类的成员初始化顺序和初始化块知识就这样回答他.下面是代码: package com.test; public class TestClass{ // 成员变量赋值第一个 ...

  9. Java类中各种静态变量的加载顺序的学习

    最近在补<thinking in java>...有一节提到了加载类需要做的一些准备...我照着书本敲了一下代码...同时稍微修改了一下书本上的代码.... package charpte ...

随机推荐

  1. 【扫盲】HTML5、Web APP、3G网站、Wap网站傻傻分不清楚

           移动互联网指手机网站和app,其中app分为native app(原生app),web app(html5开发),Hybrid app(前两种app结合):手机网站分为wap网站和3g网 ...

  2. Jquery选择器 选择一个不存在的元素 为什么不会返回 false

    不管找没找到,$()函数都会返回一个jquery对象,这个jquery对象有个length属性,表示找到多少个匹配的DOM元素,为0就是没找到.

  3. [性能测试]:关于MQ协议脚本开发

    消息队列(MQ)是一种应用程序对应用程序的通信方法.应用程序通过写和检索出入列队的针对应用程序的数据(消息)来通信,而无需专用连接来链接它们. 银行脚本使用MQ通信的较多,下面介绍一个MQ的脚本: M ...

  4. (四)Audio子系统之AudioRecord.read

      在上一篇文章<(三)Audio子系统之AudioRecord.startRecording>中已经介绍了AudioRecord如何开始录制音频,接下来,继续分析AudioRecord方 ...

  5. destoon调用方法汇总 ---转载

    根目录.模板目录和样式目录:{DT_PATH}{DT_SKIN}导入头脚:{template 'header'}{template 'footer'}对应模块首页:{$MODULE[$moduleid ...

  6. 五一,期待一场这样的旅行,提前预祝Csdner五一快乐

    五一,期待一场这样的旅行,提前预祝Csdner五一快乐 五一,你是否期待一次这样的旅行: 住在一间安静优美的小屋,在鸟鸣中起床,推窗有花香铺面而来.早餐过后,在阳光温暖的抚摸里,骑车踏青或光脚奔跑. ...

  7. UTF8最好不要带BOM,附许多经典评论

    UTF-8 不需要 BOM,尽管 Unicode 标准允许在 UTF-8 中使用 BOM.所以不含 BOM 的 UTF-8 才是标准形式,在 UTF-8 文件中放置 BOM 主要是微软的习惯(顺便提一 ...

  8. 配置MySQL接受远程登录连接

    一 开放mysql mysql的配置文件在/etc/mysql/my.cnf文件内,里面有一行bind-address = 127.0.0.1表示只允许本地访问,将这行注释即可 # bind-addr ...

  9. 深度学习(十六) ReLU为什么比Sigmoid效果好

    sigmoid: Relu: 为什么通常Relu比sigmoid和tanh强,有什么不同?主要是因为它们gradient特性不同. 1.sigmoid和tanh的gradient在饱和区域非常平缓,接 ...

  10. [中英对照]Why Redis beats Memcached for caching | 在cache化方面,为何Redis胜过Memcached?

    对Memcached和Redis有兴趣的同学不妨花几分钟读一读本文,否则请飘过. Why Redis beats Memcached for caching | 在cache化方面,为何Redis胜过 ...