1、Hibernate中Session线程安全问题:
1)、Hibernate中的session并不是HttpSession,可以理解为基于JDBC的Connection,对象生命周期、事务的管理、以及数据的读取都与Session息息相关。 2)、Session并不是线程安全的,如果多个线程同时使用一个Session来进行数据的存取,可想而知会导致数据混乱(张三新增一条数据,李四提交事务,王五还没来得及删除...)。 3)、好了,ThreadLocal该出场了。
2、ThreadLocal是什么:
1)、ThreadLocal它不是一个线程,而是线程的一个本地化对象,看源码上写的注释是:This class provides thread-local variables,这个类提供线程局部变量,如果有一个对象在多线程环境下被多个线程同时使用,可以用ThreadLocal来维护这个对象,ThreadLocal会为每个使用到该对象的线程分配一个独立的对象副本,而且线程与线程之间互不影响,各自维护自己的副本。 2)、像之前Hibernate中的Session一样,当多个请求过来需要开多个线程来获取Session,Hibernate是从SessionFactory是来获取Session的,这里就可以用ThreadLocal来维护Session这个变量,来一个请求(线程)就拷贝一个Session存入到当前线程的ThreadLocalMap这个map中,这个map是以当前ThreadLocal对象为key,Session变量为value来存的。 3)、下面的threadSession维护多个线程需要的资源:Session,先通过get()方法来查询当前这个线程里的ThreadLocal对象里的数组中是否存在Session,不存在,则创建一个新的Sessio,然后通过set(s)方法把这个Session和当前线程绑定在一起,其实每个线程都有个ThreadLocal,都是上面那个共享的threadSession,而这个对象里面的一个数组维护了上述所有线程绑定的Session,key值均为threadSession对象,而找到指定线程绑定的Session就调用到ThreadLocal对象的nextHashCode()方法,返回的是个哈希值,再和当前数组长度相与,就是该线程存放在ThreadLocal数组中的具体位置,这样就能取出指定线程绑定的指定Session了。 private static final ThreadLocal threadSession = new ThreadLocal(); public static Session getSession() throws InfrastructureException { //先判断这个线程是否有Session Session s = (Session) threadSession.get(); try { if (s == null) { //没有,则创建一个Session放入到当前线程的map中 s = getSessionFactory().openSession(); threadSession.set(s); } } catch (HibernateException ex) { throw new InfrastructureException(ex); } //有的话直接返回 return s; }
3、get()方法源码:
public T get() { // 当前线程 Thread t = Thread.currentThread(); // ThreadLocalMap对象是Thread类中的一个属性 // ThreadLocal.ThreadLocalMap threadLocals = null; // 这里就是获取当前线程中的threadLocals这个map ThreadLocalMap map = getMap(t); if (map != null) return (T) map.get(this); // 局部变量的默认初始值 T value = initialValue(); // 给当前线程创建map存放局部变量 createMap(t, value); return value; } ThreadLocalMap getMap(Thread t) { // 获取线程存放局部变量的map return t.threadLocals; } //根据当前ThreadLocal来获取变量值 private Object get(ThreadLocal key) { int i = key.threadLocalHashCode & (table.length - 1); Entry e = table[i]; if (e != null && e.get() == key) return e.value; return getAfterMiss(key, i, e); } void createMap(Thread t, T firstValue) { // 给当前线程中的存放线程局部变量的map赋值,this为ThreadLocal,值为需要存放的变量 t.threadLocals = new ThreadLocalMap(this, firstValue); }
4、set()方法源码:
public void set(T value) { //当前线程 Thread t = Thread.currentThread(); //获取当前线程的ThreadLocalMap threadLocals这个属性的值 ThreadLocalMap map = getMap(t); if (map != null) map.set(this, value); else createMap(t, value); }
5、remove()方法源码:
public void remove() { //获取当前线程的map ThreadLocalMap m = getMap(Thread.currentThread()); if (m != null) //Removes the value for this ThreadLocal m.remove(this); }
6、小结:
1)、Thread类中有一个成员变量是ThreadLocalMap类型,它是一个Map,它的key为ThreadLocal类。 2)、每个线程都拥有一个用ThreadLocal来维护的变量,各个线程之间互不影响,人手一个。 3)、变量的生命周期由线程决定,开始于第一次set()/get()方法。 4)、每个线程拥有一个自己的变量,变量就存放在线程的ThreadLocalMap中,并非共享或拷贝。
参考自: http://www.iteye.com/topic/777716
http://www.iteye.com/topic/179040
相关推荐
**线程局部变量(ThreadLocal)是Java编程中一个非常重要的工具类,它在多线程环境下提供了线程安全的数据存储。ThreadLocal并不是一个变量,而是一个类,它为每个线程都创建了一个独立的变量副本,使得每个线程都...
在 `LeakingServlet` 的 `doGet` 方法中,如果 `ThreadLocal` 没有设置值,那么会创建一个新的 `MyCounter` 并设置到 `ThreadLocal` 中。关键在于,一旦 `MyCounter` 被设置到 `ThreadLocal`,那么它将与当前线程...
Java事务和ThreadLocal是两种在Java编程中至关重要的概念,它们分别用于处理多线程环境下的数据一致性问题和提供线程局部变量。 首先,我们来深入理解Java事务。在数据库操作中,事务是一系列操作的集合,这些操作...
### 正确理解ThreadLocal:深入解析其工作原理与应用场景 #### 一、ThreadLocal的基本概念 `ThreadLocal`是Java平台提供的一种线程局部变量的解决方案,它为每一个使用该变量的线程都提供了独立的变量副本,使得每...
ThreadLocal是Java编程中一种非常特殊的变量类型,它主要用于在多线程环境下为每个线程提供独立的变量副本,从而避免了线程间的数据共享和冲突。然而,ThreadLocal在理解和使用过程中容易产生一些误区,这里我们将...
ThreadLocal是Java编程语言中的一个线程局部变量类,它为每个线程提供了一个独立的变量副本,使得每个线程可以独立地改变自己的副本,而不会影响其他线程所对应的副本。这个特性在多线程环境下处理并发问题时非常...
### Java中ThreadLocal详解 #### 一、ThreadLocal概述 在Java多线程编程中,`ThreadLocal`是一个非常重要的工具类,它提供了一种在每个线程内部存储线程私有实例的方法。通常情况下,当多个线程共享某个变量时,...
ThreadLocal 整理 ThreadLocal 是 Java 中的一个重要组件,它能够在每个线程中保持独立的副本。这个功能是通过 Thread 类中的 threadLocals 属性来实现的,这个属性实际上是一个 Entry 数组,其中的每个 Entry 都...
private static final ThreadLocal<Connection> connectionHolder = new ThreadLocal(); public static void setConnection(Connection conn) { connectionHolder.set(conn); } public static Connection get...
Java中的ThreadLocal是一个非常重要的工具类,它在多线程编程中扮演着独特角色,尤其在处理线程间数据隔离和共享时。ThreadLocal不是线程本身,而是为每个线程提供一个独立的变量副本,使得每个线程都可以独立地改变...
本资料主要聚焦于两种设计模式以及Java中的ThreadLocal特性。 首先,我们来探讨单例模式。单例模式是一种确保一个类只有一个实例,并提供全局访问点的设计模式。在Java中,通常通过私有构造函数、静态工厂方法或...
**线程局部变量(ThreadLocal)是Java编程中一个非常重要的概念,主要用于在多线程环境中为每个线程提供独立的变量副本。ThreadLocal不是一种数据结构,而是一种解决线程间共享数据的方式,它提供了线程安全的局部...
在Java编程中,ThreadLocal是线程局部变量的类,它提供了一种在多线程环境中为每个线程创建和维护独立副本的机制。ThreadLocal主要用于解决线程间的数据隔离问题,确保各线程拥有自己的变量副本,避免了数据共享带来...
Java中的`ThreadLocal`类是一个非常实用的工具,它提供了线程局部变量的功能。线程局部变量意味着每个线程都拥有自己独立的变量副本,互不干扰,这在多线程编程中尤其有用,可以避免数据共享带来的同步问题。下面...
这个压缩包 "Quartz-ThreadLocal.rar" 内含的学习资源很可能是关于如何在 Quartz 调度器中结合使用 ThreadLocal 的示例。 Quartz 的核心功能包括: 1. **作业与触发器**:在 Quartz 中,任务被称为“作业”(Job)...
**标题:“JDK的ThreadLocal理解(一)使用和测试”** **正文:** ThreadLocal是Java中的一个非常重要的线程安全工具类,它在多线程编程中扮演着独特的角色。通过创建ThreadLocal实例,我们可以为每个线程提供一个...
Java中的ThreadLocal是一个非常重要的工具类,它在多线程编程中扮演着独特角色,用于为每个线程提供独立的变量副本。理解ThreadLocal的工作原理和使用方法对于编写高效、安全的多线程程序至关重要。 ### ...