前言:工作中将要使用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. Python(序列化json,pickle,shelve)

    序列化 参考:https://www.cnblogs.com/yuanchenqi/articles/5732581.html # dic = str({'1':'111'}) # # f = ope ...

  2. ionic 项目 随笔

    1,首先 会进入src/index.html, <!-- The polyfills js is generated during the build process --> <sc ...

  3. 由button标签在 IE 8.0 下的异常表现引发的一场血案

    写在最前的最后:整篇文章絮絮叨叨说了半天,我得出一个最佳实践:和button标签say goodbay,用 a 标签模拟之. 首先看一个在chrome 下的简单demo 这样的布局在组件开发中再常见不 ...

  4. SpringBoot idea maven打包war及运行war包

    pom.xml修改打包类型pom改为war <artifactId>Test02</artifactId> <packaging>war</packaging ...

  5. 关于找不到stdafx.h头文件问题(pass)

    代码: #include "stdafx.h" #include "stdlib.h" char* getcharBuffer() { return " ...

  6. python-----读写操作

    1. 文件的读取 注意:在windows中文件的路径是这样:C:\Users\name\mystuff .由于\u在python中表示转义如果使用此文件路径就会报错.解决方法: a.把斜杠\改为反斜杠 ...

  7. java_I/O字节流

    I/O流(Stream) INPUT:输入流,从文件里读OUPUT:输出流,写内容到文件 IO流分为:字符流和字节流 字符流:处理纯文本文件. 字节流:处理可以所有文件. 测试字节输出流OuPut(写 ...

  8. DataTables复杂表头

    工作上的需要,要做一个复杂的表头的DataTables thead如下 遇到的问题(详细问题可以浏览官网的答案 链接) 需自定义表头(thead),如果不自定义则会 Cannot read prope ...

  9. 认识python正则模块re

    python正则模块re python中re中内置匹配.搜索.替换方法见博客---python附录-re.py模块源码(含re官方文档链接) 正则的应用是处理一些字符串,phthon的博文python ...

  10. net与树莓派的情缘-安装与卸载MySql(五)

    安装MySql sudo apt-get install mysql-server 删除 mysql sudo apt-get autoremove --purge mysql-server-5.0s ...