ThreadLocal(二)

2014-11-24 03:19:40 · 作者: · 浏览: 3
dLocals,它是把ThreadLocal作为threadLocals键值在使用的。这点比示例1的算法先进多。它没有了同步问题。因为它被创建后根本就没读写操作。

注意2:ThreadLocalMap就在ThreadLocal中,但是它并没有使用HashMap,它使用的算法有复杂,没看明白。

注意3:如果我们对hashcode不是通过该对象的成员生成,而是使其自动生成时,且采用取余形式得到哈希值,增量(HASH_INCREMENT)最好是个素数。这样增量的才不能被任何才小于取余值(哈希表的长度)整除,否则会哈希到同一个桶中。

ThreadLocalMap中就是对hashcode取余的方式得到哈希值的。

int len = tab.length;

int i = key.hashCode & (len-1);

因为

/**

* The initial capacity -- MUST be a power of two.

*/

private static final int INITIAL_CAPACITY = 16;

所以上面的"int i = key.hashCode & (len-1);"相当于"int i = key.hashCode %len;"

虽然"HASH_INCREMENT"不是个素数,但是它却不能被tab.length(它等于2的n此方且大于等于16)被整除的。

注意4:ThreadLocalMap中采用的是弱引用

static class Entry extends WeakReference {

/** The value associated with this ThreadLocal. */

Object value;

Entry(ThreadLocal k, Object v) {

super(k);

value = v;

}

}

注意5:JDK为什么不使用hashMap或WeakHashMap,而是自己写了ThreadLocalMap。应该主要是为提供了对不同key(这里是ThreadLocal)对象的但hashCode相同的存储的支持。

如果希望线程局部变量初始化其它值,那么需要自己实现ThreadLocal的子类并重写initialValue()该方法,

通常使用一个内部匿名类对ThreadLocal进行子类化,比如下面的例子,SerialNum类为每一个类分配一个序号:

Java代码

public class SerialNum { :

// The next serial number to be assigned

private static int nextSerialNum = 0;

private static ThreadLocal serialNum = new ThreadLocal() {

protected synchronized Object initialValue() {

return new Integer(nextSerialNum++);

}

};

public static int get() {

return ((Integer) (serialNum.get())).intValue();

}

}

SerialNum类的使用将非常地简单,因为get()方法是static的,所以在需要获取当前线程的序号时,简单地调用:

Java代码

int serial = SerialNum.get();

即可。

使用方法一

Hibernate的文档中关于使ThreadLocal管理多线程访问的部分。

具体代码如下

public static final ThreadLocal session = new ThreadLocal();

public static Session currentSession() {

Session s = (Session)session.get();

//open a new session,if this session has none

if(s == null){

s = sessionFactory.openSession();

session.set(s);

}

return s;

}

我们逐行分析

1。 初始化一个ThreadLocal对象,ThreadLocal有三个成员方法get()、set()、initialvalue()。

如果不初始化initialvalue,则initialvalue返回null。

3。session的get根据当前线程返回其对应的线程内部变量,也就是我们需要的net.sf.hibernate.Session(相当于对应每个数据库连接).

多线程情况下共享数据库链接是不安全的。ThreadLocal保证了每个线程都有自己的s(数据库连接)。

5。如果是该线程初次访问,自然,s(数据库连接)会是null,接着创建一个Session,具体就是行6。

6。创建一个数据库连接实例s

7。保存该数据库连接s到ThreadLocal中。

8。如果当前线程已经访问过数据库了,则从session中get()就可以获取该线程上次获取过的连接实例。

使用方法二

当要给线程初始化一个特殊值时,需要自己实现ThreadLocal的子类并重写该方法,

通常使用一个内部匿名类对ThreadLocal进行子类化,EasyDBO中创建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);

jdbcContext.set(context);

}

return context;

}

private static class JDBCContextThreadLocal extends ThreadLo