什么是ThreadLocal?
顧名思義它是local variable(線(xiàn)程局部變量)。它的功用非常簡(jiǎn)單,就是為每一個(gè)使用該變量的線(xiàn)程都提供一個(gè)變量值的副本,是每一個(gè)線(xiàn)程都可以獨立地改變自己的副本,而不會(huì )和其它線(xiàn)程的副本沖突。從線(xiàn)程的角度看,就好像每一個(gè)線(xiàn)程都完全擁有該變量。
使用場(chǎng)景
ThreadLocal類(lèi)
它主要由四個(gè)方法組成initialValue(),get(),set(T),remove(),其中值得注意的是initialValue(),該方法是一個(gè)protected的方法,顯然是為了子類(lèi)重寫(xiě)而特意實(shí)現的。該方法返回當前線(xiàn)程在該線(xiàn)程局部變量的初始值,這個(gè)方法是一個(gè)延遲調用方法,在一個(gè)線(xiàn)程第1次調用get()或者set(Object)時(shí)才執行,并且僅執行1次。ThreadLocal中的確實(shí)實(shí)現直接返回一個(gè)null:
ThreadLocal的原理
ThreadLocal是如何做到為每一個(gè)線(xiàn)程維護變量的副本的呢?其實(shí)實(shí)現的思路很簡(jiǎn)單,在ThreadLocal類(lèi)中有一個(gè)Map,用于存儲每一個(gè)線(xiàn)程的變量的副本。比如下面的示例實(shí)現:
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的文檔時(shí)看到了關(guān)于使ThreadLocal管理多線(xiàn)程訪(fǎng)問(wèn)的部分。具體代碼如下
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。 初始化一個(gè)ThreadLocal對象,ThreadLocal有三個(gè)成員方法 get()、set()、initialvalue()。
如果不初始化initialvalue,則initialvalue返回null。
3。session的get根據當前線(xiàn)程返回其對應的線(xiàn)程內部變量,也就是我們需要的net.sf.hibernate.Session(相當于對應每個(gè)數據庫連接).多線(xiàn)程情況下共享數據庫鏈接是不安全的。ThreadLocal保證了每個(gè)線(xiàn)程都有自己的s(數據庫連接)。
5。如果是該線(xiàn)程初次訪(fǎng)問(wèn),自然,s(數據庫連接)會(huì )是null,接著(zhù)創(chuàng )建一個(gè)Session,具體就是行6。
6。創(chuàng )建一個(gè)數據庫連接實(shí)例 s
7。保存該數據庫連接s到ThreadLocal中。
8。如果當前線(xiàn)程已經(jīng)訪(fǎng)問(wèn)過(guò)數據庫了,則從session中g(shù)et()就可以獲取該線(xiàn)程上次獲取過(guò)的連接實(shí)例。
使用方法二
當要給線(xiàn)程初始化一個(gè)特殊值時(shí),需要自己實(shí)現ThreadLocal的子類(lèi)并重寫(xiě)該方法,通常使用一個(gè)內部匿名類(lèi)對ThreadLocal進(jìn)行子類(lèi)化,EasyDBO中創(chuàng )建jdbc連接上下文就是這樣做的:
public class JDBCContext{
private static Logger logger = Logger.getLogger(JDBCContext.class);
private DataSource ds;
protected Connection connection;
private boolean isValid = true;
private static ThreadLocal jdbcContext;
private JDBCContext(DataSource ds){
this.ds = ds;
createConnection();
}
public static JDBCContext getJdbcContext(javax.sql.DataSource ds)
{
if(jdbcContext==null)jdbcContext=new JDBCContextThreadLocal(ds);
JDBCContext context = (JDBCContext) jdbcContext.get();
if (context == null) {
context = new JDBCContext(ds);
}
return context;
}
private static class JDBCContextThreadLocal extends ThreadLocal {
public javax.sql.DataSource ds;
public JDBCContextThreadLocal(javax.sql.DataSource ds)
{
this.ds=ds;
}
protected synchronized Object initialValue() {
return new JDBCContext(ds);
}
}
}
使用單例模式,不同的線(xiàn)程調用getJdbcContext()獲得自己的jdbcContext,都是通過(guò)JDBCContextThreadLocal 內置子類(lèi)來(lái)獲得JDBCContext對象的線(xiàn)程局部變量,這個(gè)變
聯(lián)系客服