什么是ThreadLocal?

顧名思義它是local variable(線程局部變量)。它的功用非常簡單,就是為每一個使用該變量的線程都提供一個變量值的副本,是每一個線程都可以獨立地改變自己的副本,而不會和其它線程的副本沖突。從線程的角度看,就好像每一個線程都完全擁有該變量。

使用場景

  1. To keep state with a thread (user-id, transaction-id, logging-id)
  2. To cache objects which you need frequently

ThreadLocal類

它 主要由四個方法組成initialValue(),get(),set(T),remove(),其中值得注意的是initialValue(),該方法 是一個protected的方法,顯然是為了子類重寫而特意實現(xiàn)的。該方法返回當前線程在該線程局部變量的初始值,這個方法是一個延遲調(diào)用方法,在一個線 程第1次調(diào)用get()或者set(Object)時才執(zhí)行,并且僅執(zhí)行1次。ThreadLocal中的確實實現(xiàn)直接返回一個null:

ThreadLocal的原理

ThreadLocal是如何做到為每一個線程維護變量的副本的呢?其實實現(xiàn)的思路很簡單,在ThreadLocal類中有一個Map,用于存儲每一個線程的變量的副本。比如下面的示例實現(xiàn):

public class ThreadLocal
{
private Map values = Collections.synchronizedMap(new HashMap());
public Object get()
{
Thread curThread = Thread.currentThread();
Object o = values.get(curThread);
if (o == null && !values.containsKey(curThread))
{
o = initialValue();
values.put(curThread, o);
}
return o;
}

 public void set(Object newValue)
{
values.put(Thread.currentThread(), newValue);
}

 public Object initialValue()
{
return null;
}
}

ThreadLocal 的使用

使用方法一:

Hibernate的文檔時看到了關于使ThreadLocal管理多線程訪問的部分。具體代碼如下

1.  public static final ThreadLocal session = new ThreadLocal();
2.  public static Session currentSession() {
3.      Session s = (Session)session.get();
4.      //open a new session,if this session has none
5.   if(s == null){
6.      s = sessionFactory.openSession();
7.      session.set(s);
8.   }
      return s;
9. }

我們逐行分析
1。 初始化一個ThreadLocal對象,ThreadLocal有三個成員方法 get()、set()、initialvalue()。
    如果不初始化initialvalue,則initialvalue返回null。
3。 session的get根據(jù)當前線程返回其對應的線程內(nèi)部變量,也就是我們需要的net.sf.hibernate.Session(相當于對應每個數(shù)據(jù) 庫連接).多線程情況下共享數(shù)據(jù)庫鏈接是不安全的。ThreadLocal保證了每個線程都有自己的s(數(shù)據(jù)庫連接)。
5。如果是該線程初次訪問,自然,s(數(shù)據(jù)庫連接)會是null,接著創(chuàng)建一個Session,具體就是行6。
6。創(chuàng)建一個數(shù)據(jù)庫連接實例 s
7。保存該數(shù)據(jù)庫連接s到ThreadLocal中。
8。如果當前線程已經(jīng)訪問過數(shù)據(jù)庫了,則從session中get()就可以獲取該線程上次獲取過的連接實例



以上轉(zhuǎn)自 橄欖樹  http://hi.baidu.com/mojiedao/blog/item/0ceeab99e7a0ad086f068c09.html