ThreadLocal浅析

原写于2010-12-11

 

Java并发,对于共享资源的控制,是一个复杂的事情,比如:同步等。 ThreadLocal 为这个问题提供另外一种解决方案,其实根本的原理用空间换时间

 

一、结构图

ThreadLocal浅析_第1张图片

 

二、源码阅读

关注切入点set()get()方法

 

1.set()方法

ThreadLocal浅析_第2张图片

 

结论:对ThreadLocal的操作,实际委托给当前Thread,每个Thread都会有自己独立的ThreadLocalMap实例,存储的仓库是Entry[] tableEntrykeyThreadLocalvalue为存储内容;因此在并发环境下,对ThreadLocalsetget,不会有任何问题。

 

2.get方法

ThreadLocal浅析_第3张图片

 

遗留问题

1.Thread.threadLocals 什么时候实例化线程实例化时吗?答:第一次set时,会判断是否为null,若为null,初始化。

2.ThreadLocalMap.replaceStaleEntry(key, value, i); 做了什么全清洗stale对象;存放当前对象在发现的第一个stale位置。

3.ThreadLocalMap.getEntryAfterMiss(ThreadLocal key, int i, Entry e)作用

查找没有存放在hash计算出index位置的元素(为什么出现此情况?如果发现已经被gc的内容时,会直接把当前对象存放在已被gc对象的位置)此做法,与Map方式,有区别Map策略:a、相同hash&key,覆盖value b、相同hashkey不同,当前元素做为单向链的第一个元素,原来第一个元素做为当前元素的下一个。ThreadLocalMap原理,是不存在b情况,以ThreadLocal作为keyThreadLocal threadLocalHashCode 属性唯一的标识一个实例,Entry[]数组的偏移,是通过threadLocalHashCode & (length-1)计算,因为不存在:threadLocalHashCode相同,但取出存方在Entrykey不同(keyThreadLocal实例).

 

三、应用场景

 

public class SqlMapClientImpl {
  protected ThreadLocal localSqlMapSession = new ThreadLocal();
  public int delete(String id) throws SQLException {
    return getLocalSqlMapSession().delete(id);  //所有操作,首先getLocalSqlMapSession()获取SqlMapSessionImpl
  }
  protected SqlMapSessionImpl getLocalSqlMapSession() {
    SqlMapSessionImpl sqlMapSession = (SqlMapSessionImpl) localSqlMapSession.get(); //从ThreadLocal获取SqlMapSessionImpl
    if (sqlMapSession == null || sqlMapSession.isClosed()) {
      sqlMapSession = new SqlMapSessionImpl(this);
      localSqlMapSession.set(sqlMapSession); //存入ThreadLocal
    }
    return sqlMapSession;
  }
}

 

结论:当SqlMapClient为单实例时,多个请求操作,也就是多个线程操作,此时获取的SqlMapSessionImpl是针对当前线程的;在servlet容器,使用一个线程池服务请求,所以SqlMapSessionImpl是能被复用,提高效率。

 

 

 

你可能感兴趣的:(技术点滴)