`
aladdin_leon
  • 浏览: 117269 次
  • 性别: Icon_minigender_1
  • 来自: 哈尔滨
社区版块
存档分类
最新评论

利于ThreadLocal管理Hibernate Session

阅读更多

      在利用Hibernate开发DAO模块时,我们和Session打的交道最多,所以如何合理的管理Session,避免Session的频繁创建和销毁,对于提高系统的性能来说是非常重要的,以往是通过eclipse的插件来自动完成这些代码的,当然效果是不错的,但是总是觉得不爽(没有读懂那些冗长的代码),所以现在打算自己实现Session管理的代码。我们知道Session是由SessionFactory负责创建的,而SessionFactory的实现是线程安全的,多个并发的线程可以同时访问一个SessionFactory并从中获取Session实例,那么Session是否是线程安全的呢?很遗憾,答案是否定的。Session中包含了数据库操作相关的状态信息,那么说如果多个线程同时使用一个Session实例进行CRUD,就很有可能导致数据存取的混乱,你能够想像那些你根本不能预测执行顺序的线程对你的一条记录进行操作的情形吗?
      在Session的众多管理方案中,我们今天来认识一种名为ThreadLocal模式的解决方案。
      早在Java1.2推出之时,Java平台中就引入了一个新的支持:java.lang.ThreadLocal,给我们在编写多线程程序时提供了一种新的选择。ThreadLocal是什么呢?其实ThreadLocal并非是一个线程的本地实现版本,它并不是一个Thread,而是thread local variable(线程局部变量)。也许把它命名为ThreadLocalVar更加合适。线程局部变量(ThreadLocal)其实的功用非常简单,就是为每一个使用该变量的线程都提供一个变量值的副本,是每一个线程都可以独立地改变自己的副本,而不会和其它线程的副本冲突。从线程的角度看,就好像每一个线程都完全拥有一个该变量。
      ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单,在ThreadLocal类中有一个Map,用于存储每一个线程的变量的副本。比如下面的示例实现(为了简单,没有考虑集合的泛型):

  1. public class ThreadLocal {   
  2.      private Map values = Collections.synchronizedMap(new HashMap());   
  3.      public Object get() {   
  4.         Thread currentThread = Thread.currentThread();    
  5.         Object result = values.get(currentThread);    
  6.         if(result == null&&!values.containsKey(currentThread)) {   
  7.            result = initialValue();   
  8.            values.put(currentThread, result);    
  9.         }   
  10.         return result;    
  11.      }   
  12.      public void set(Object newValue) {   
  13.         values.put(Thread.currentThread(), newValue);   
  14.      }   
  15.      public Object initialValue() {   
  16.         return null;    
  17.      }   
  18. }   

      那麽具体如何利用ThreadLocal来管理Session呢?Hibernate官方文档手册的示例之中,提供了一个通过ThreadLocal维护Session的好榜样:

  1. public class HibernateUtil {   
  2.        public static final SessionFactory sessionFactory;   
  3.        static {   
  4.            try {   
  5.                  sessionFactory = new Configuration().configure()   
  6.                                    .buildSessionFactory();   
  7.            } catch (Throwable ex) {   
  8.                 throw new ExceptionInInitializerError(ex);   
  9.            }   
  10.        }   
  11.        public static final ThreadLocal<session></session> session =    
  12.                                             new ThreadLocal<session></session>();   
  13.        public static Session currentSession() throws HibernateException {   
  14.                Session s = session.get();   
  15.                if(s == null) {   
  16.                      s = sessionFactory.openSession();   
  17.                      session.set(s);   
  18.                }   
  19.                return s;   
  20.        }   
  21.        public static void closeSession() throws HibernateException {   
  22.                 Session s = session.get();   
  23.                 if(s != null) {   
  24.                       s.close();   
  25.                 }   
  26.                 session.set(null);   
  27.        }   
  28. }  

     只要借助上面的工具类获取Session实例,我们就可以实现线程范围内的Session共享,从而避免了线程中频繁的创建和销毁Session实例。当然,不要忘记在用完后关闭Session。
     写到这里,想再多说一些,也许大多数时候我们的DAO并不会涉及到多线程的情形,比如我们不会将DAO的代码写在Servlet之中,那样不是良好的设计,我自己通常会在service层的代码里访问DAO的方法。但是我还是建议采用以上的工具类来管理Session,毕竟我们不能仅仅考虑今天为自己做什么,还应该考虑明天为自己做什么!

分享到:
评论
9 楼 huihai 2011-06-01  
我在一个java企业级应用中,用到了ThreadLocal,也是通过你上面的样子得到session。我想部一下,我的一个线程一直在循环运行,那么就会一直得到同一个session,我想问一下session会不会失效?
8 楼 h521999 2008-09-02  
感谢,对ThreadLocal有了更深的了解!
7 楼 wf_chn 2008-07-29  
可以设置current_session_context_class参数
6 楼 javapupil 2008-07-24  
再进一步说说:
每个Thread一个Session,在设计中要慎用Lazy Load。因为一个Bean(Hibernate Entity)一旦是Lazy Load,同时该Bean的生命周期超过创建该Bean的线程,或者两个以上线程共享该Bean时,就会造成抛出HibernateException,告诉你Session已经被Close。
5 楼 javapupil 2008-07-24  
要解决这个问题,有两种做法:
1. 很蠢的方法:随时对Session进行Close。之所以说蠢,是因为这样完全失去了用ThreadLocal共享Session的意义。
2. 我目前用的方法:给Servlet加上Filter,在Filter中,对ThreadLocal中的Session进行Close操作。通过这样做,保证在一个Servlet请求的生命周期中,Session能复用。下一个Servlet请求即使和上一个复用同一个线程,也会新建Session。
   这种做法,其实把Session的复用周期从“整个线程复用”减少到了“整个ServletRequest期间复用”

其实可能还有一种方式:在同一线程,且同一Http Session范围内复用Hibernate Session。这个我没有尝试,因为我做的这个项目的特点不适合这么做。但我想也是可行的。感兴趣的可以自己做一做。
4 楼 javapupil 2008-07-24  
此类分析,网上很多很多。但大家都忽略了一个问题:
应用服务器往往内部使用线程池。这会造成一个线程并不随着Servlet请求完成而结束。那么下一个Servlet请求就会复用这个线程的Session。后果是张三登陆后看到了李四的数据。
怎么解决? ==(离开一下)
3 楼 BigBlue 2008-05-14  
这种用法一些情况下有问题,需要慎重
http://www.hibernate.org/207.html
2 楼 endeavor416 2008-04-16  
  ,看完你的这篇文章之后,我的问题也解决了!谢谢楼主的文章
1 楼 Virgo_S 2008-04-02  
不错,正好,正要用

相关推荐

Global site tag (gtag.js) - Google Analytics