前言:工作中将要使用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. lamp-linux2

    LAMP编程之Linux(2) 一.进阶指令 1.du指令 作用:du表示directory used,显示出目录所占的磁盘空间大小的情况. 语法:#du -sh 目录路径 选项说明: -s:表示su ...

  2. 为什么说 Gumroad 是一家 “失败” 的创业公司?

    Gumroad 是一家 "失败" 的创业公司. 创立于 2012 年,Gumroad 是一个面向创造者的电商平台.创始人 Sahil Lavingia,一名 19 岁的少年,Pin ...

  3. python bytes和str之间的转换

    1 # bytes object 2 b = b"example" 3 4 # str object 5 s = "example" 6 7 # str to ...

  4. Python使用浏览器模拟访问页面之使用ip代理

    最近需要使用浏览器模拟访问页面,同时需要使用不同的ip访问,这个时候就考虑到在使用浏览器的同时加上ip代理. 本篇工作环境为win10,python3.6. Chorme 使用Chrome浏览器模拟访 ...

  5. phpspreadsheet开发手记

    坑安装简单示例通过模板来生成文件释放内存单元格根据索引获取英文列设置值合并单元格居中显示宽度设置批量设置单元格格式直接输出下载自动计算列宽函数formula单元格变可点击的超链 PhpSpreadsh ...

  6. OpenERP中自定义模块卸载失败,Postgres数据库删不掉数据库,OpenERP登录不了一直在加载的问题解决方案。

    解决方案也就是删除掉不用的数据库,OE会提示当前有N个Session不让Drop数据库. 对于Postgres 9.1 版本,在pgAdmin中查询以下语句: SELECT pg_terminate_ ...

  7. (转)[小工具] Linux下列格式化工具 - column

    当你看到Linux下命令输出的列歪歪扭扭时,是不是看着很不爽?column 命令就可以方便地解决这个问题. 比如: 我们一般就只用到上面这一个用法. column的其他用法如下: 选项 含义 -c 字 ...

  8. 快速创建SpringBoot+SSM解析

    此处使用IDEA快速搭建SpringBoot应用,首先用SpringBoot搭建WEB工程: 然后点击Next生成项目,首次生成可能有点慢,下次创建的时候就会快很多,生成后的目录结构如下: 我们更改下 ...

  9. Compile android source and kernel for emulator in Debian

    1.download the android source code Reference from http://source.android.com/source/downloading.html ...

  10. 深度学习(七)U-Net原理以及keras代码实现医学图像眼球血管分割

    原文作者:aircraft 原文链接:https://www.cnblogs.com/DOMLX/p/9780786.html DRIVE数据集下载百度云链接:链接:https://pan.baidu ...