深入了解ThreadLocal
2017-12-12 16:23
246 查看
ThreadLocal是什么
早在JDK 1.2的版本中就提供java.lang.ThreadLocal,ThreadLocal为解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序。
ThreadLocal,顾名思义,它不是一个线程,而是线程的一个本地化对象。当工作于多线程中的对象使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程分配一个独立的变量副本。所以每一个线程都可以独立地改变自己的副本,而不会影响其他线程所对应的副本。从线程的角度看,这个变量就像是线程的本地变量,这也是类名中“Local”所要表达的意思。
ThreadLocal的接口方法
ThreadLocal类接口很简单,只有4个方法
void set(Object value) 设置当前线程的线程局部变量的值;
public Object get() 该方法返回当前线程所对应的线程局部变量;
public void remove() 将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度;
protected Object initialValue() 返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的默认实现直接返回一个null。
值得一提的是,在JDK5.0中,ThreadLocal已经支持泛型,该类的类名已经变为ThreadLocal<T>。API方法也相应进行了调整,新版本的API方法分别是void set(T value)、T get()以及T initialValue()。
ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal类中有一个Map,用于存储每一个线程的变量副本,Map中元素的键为线程对象,而值对应线程的变量副本。
ThreadLocal例子
package com.ksf.server.util;
import com.ksf.server.bean.session.ManagerUserBean;
import com.ksf.server.bean.session.ShopUserBean;
import com.ksf.server.bean
9afe
.session.WeixinUserBean;
/**
* 描述:
* @author dashu.
* @date 2016/08/02.
*/
public class ReqContext {
private static final ThreadLocal<String> sessinoIdContext = new ThreadLocal<String>();
private static final ThreadLocal<ReqInfo> reqInfoContext = new ThreadLocal<ReqInfo>();
private static final ThreadLocal<String> managerTokenContext = new ThreadLocal<String>();
private static final ThreadLocal<ManagerUserBean> managerUserContext = new ThreadLocal<ManagerUserBean>();
private static final ThreadLocal<WeixinUserBean> weixinUserContext = new ThreadLocal<WeixinUserBean>();
private static final ThreadLocal<ShopUserBean> shopUserContext = new ThreadLocal<ShopUserBean>();
public static String getSessionId() {
return sessinoIdContext.get();
}
public static void setSessionId(String sessionId) {
sessinoIdContext.set(sessionId);
}
public static void removeSessionId() {
sessinoIdContext.remove();
}
public static ReqInfo getReqInfo() {
return reqInfoContext.get();
}
public static void setReqInfo(ReqInfo reqInfo) {
reqInfoContext.set(reqInfo);
}
public static void removeReqInfo() {
reqInfoContext.remove();
}
public static String getManagerToken() {
return managerTokenContext.get();
}
public static void setManagerToken(String token) {
managerTokenContext.set(token);
}
public static void removeManagerToken() {
managerTokenContext.remove();
}
public static ManagerUserBean getManagerUserBean() {
return managerUserContext.get();
}
public static void setManagerUserBean(ManagerUserBean u) {
managerUserContext.set(u);
}
public static void removeManagerUserBean() {
managerUserContext.remove();
}
public static WeixinUserBean getWeixinUserBean() {
return weixinUserContext.get();
}
public static void setWeixinUserBean(WeixinUserBean u) {
weixinUserContext.set(u);
}
public static void removeWeixinUserBean() {
weixinUserContext.remove();
}
public static ShopUserBean getShopUserBean() {
return shopUserContext.get();
}
public static void setShopUserBean(ShopUserBean u) {
shopUserContext.set(u);
}
public static void removeShopUserBean() {
shopUserContext.remove();
}
}
早在JDK 1.2的版本中就提供java.lang.ThreadLocal,ThreadLocal为解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序。
ThreadLocal,顾名思义,它不是一个线程,而是线程的一个本地化对象。当工作于多线程中的对象使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程分配一个独立的变量副本。所以每一个线程都可以独立地改变自己的副本,而不会影响其他线程所对应的副本。从线程的角度看,这个变量就像是线程的本地变量,这也是类名中“Local”所要表达的意思。
ThreadLocal的接口方法
ThreadLocal类接口很简单,只有4个方法
void set(Object value) 设置当前线程的线程局部变量的值;
public Object get() 该方法返回当前线程所对应的线程局部变量;
public void remove() 将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度;
protected Object initialValue() 返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的默认实现直接返回一个null。
值得一提的是,在JDK5.0中,ThreadLocal已经支持泛型,该类的类名已经变为ThreadLocal<T>。API方法也相应进行了调整,新版本的API方法分别是void set(T value)、T get()以及T initialValue()。
ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal类中有一个Map,用于存储每一个线程的变量副本,Map中元素的键为线程对象,而值对应线程的变量副本。
ThreadLocal例子
package com.ksf.server.util;
import com.ksf.server.bean.session.ManagerUserBean;
import com.ksf.server.bean.session.ShopUserBean;
import com.ksf.server.bean
9afe
.session.WeixinUserBean;
/**
* 描述:
* @author dashu.
* @date 2016/08/02.
*/
public class ReqContext {
private static final ThreadLocal<String> sessinoIdContext = new ThreadLocal<String>();
private static final ThreadLocal<ReqInfo> reqInfoContext = new ThreadLocal<ReqInfo>();
private static final ThreadLocal<String> managerTokenContext = new ThreadLocal<String>();
private static final ThreadLocal<ManagerUserBean> managerUserContext = new ThreadLocal<ManagerUserBean>();
private static final ThreadLocal<WeixinUserBean> weixinUserContext = new ThreadLocal<WeixinUserBean>();
private static final ThreadLocal<ShopUserBean> shopUserContext = new ThreadLocal<ShopUserBean>();
public static String getSessionId() {
return sessinoIdContext.get();
}
public static void setSessionId(String sessionId) {
sessinoIdContext.set(sessionId);
}
public static void removeSessionId() {
sessinoIdContext.remove();
}
public static ReqInfo getReqInfo() {
return reqInfoContext.get();
}
public static void setReqInfo(ReqInfo reqInfo) {
reqInfoContext.set(reqInfo);
}
public static void removeReqInfo() {
reqInfoContext.remove();
}
public static String getManagerToken() {
return managerTokenContext.get();
}
public static void setManagerToken(String token) {
managerTokenContext.set(token);
}
public static void removeManagerToken() {
managerTokenContext.remove();
}
public static ManagerUserBean getManagerUserBean() {
return managerUserContext.get();
}
public static void setManagerUserBean(ManagerUserBean u) {
managerUserContext.set(u);
}
public static void removeManagerUserBean() {
managerUserContext.remove();
}
public static WeixinUserBean getWeixinUserBean() {
return weixinUserContext.get();
}
public static void setWeixinUserBean(WeixinUserBean u) {
weixinUserContext.set(u);
}
public static void removeWeixinUserBean() {
weixinUserContext.remove();
}
public static ShopUserBean getShopUserBean() {
return shopUserContext.get();
}
public static void setShopUserBean(ShopUserBean u) {
shopUserContext.set(u);
}
public static void removeShopUserBean() {
shopUserContext.remove();
}
}
相关文章推荐
- ThreadLocal 深入了解
- 深入了解ThreadLocal
- 电容深入了解
- 原 深入了解tcmalloc(一):windows环境下无缝拦截技术初探
- 深入了解windows句柄
- 指针+函数+数组 深入了解
- 【C++】模版的概念、使用方法和深入了解
- Mybatis深入了解(七)----延迟加载
- 深入了解ConcurrentHashMap的底层实现
- 《深入了解mybatis原则》 MyBatis架构设计和案例研究
- 深入了解ApusicAS服务器配置系列之——配置Web上下文根
- 深入了解标记-清扫回收算法
- 熬之滴水成石:最想深入了解的内容--windows内核机制(6)
- 深入了解UART
- 熬之滴水成石:最想深入了解的内容--windows内核机制(7)
- Android视图绘制流程完全解析,带你一步步深入了解View(二)
- 深入了解回滚表空间丢失的解决方法
- 深入了解HBASE架构
- 深入理解ThreadLocal
- Android LayoutInflater原理分析,带你一步步深入了解View(一)