hibernate 的SessionFactory的getCurrentSession 與 openSession() 的區別

來源:互聯網
上載者:User

標籤:thread   net   方法   cto   並發   uil   get   生命週期   XML   

1 getCurrentSession建立的session會和綁定到當前線程,而openSession不會。

2 getCurrentSession建立的線程會在交易回復或事物提交後自動關閉,而openSession必須手動關閉

這裡getCurrentSession本地事務(本地事務:jdbc)時 要在設定檔裡進行如下設定

    * 如果使用的是本地事務(jdbc事務) 
<property name="hibernate.current_session_context_class">thread</property> 
* 如果使用的是全域事務(jta事務) 
<property name="hibernate.current_session_context_class">jta</property> 

getCurrentSession () 使用當前的session 
openSession()         重建立立一個新的session

在一個應用程式中,如果DAO 層使用Spring 的hibernate 模板,通過Spring 來控制session 的生命週期,則首選getCurrentSession ()。

使用Hibernate的大多數應用程式需要某種形式的“上下文相關的” session,特定的session在整個特定的上下文範圍內始終有效。然而,對不同類型的應用程式而言,要為什麼是組成這種“上下文”下一個定義通常 是困難的;不同的上下文對“當前”這個概念定義了不同的範圍。在3.0版本之前,使用Hibernate的程式要麼採用自行編寫的基於 ThreadLocal的上下文session,要麼採用HibernateUtil這樣的輔助類,要麼採用第三方架構(比如Spring或Pico), 它們提供了基於代理(proxy)或者基於攔截器(interception)的上下文相關session。

從3.0.1版本開 始,Hibernate增加了SessionFactory.getCurrentSession()方法。一開始,它假定了採用JTA事務,JTA事務 定義了當前session的範圍和上下文(scope and context)。HibernateTeam Dev堅信,因為有好幾個獨立的JTA TransactionManager實現穩定可用,不論是否被部署到一個J2EE容器中,大多數(假若不是所有的)應用程式都應該採用JTA交易管理。 基於這一點,採用JTA的上下文相關session可以滿足你一切需要。

更好的是,從3.1開 始,SessionFactory.getCurrentSession()的後台實現是可拔插的。因此,我們引入了新的擴充介面 (org.hibernate.context.CurrentSessionContext)和新的配置參數 (hibernate.current_session_context_class),以便對什麼是“當前session”的範圍和上下文(scope and context)的定義進行拔插。

請參閱 org.hibernate.context.CurrentSessionContext介面的Javadoc,那裡有關於它的契約的詳細討論。它定義 了單一的方法,currentSession(),特定的實現用它來負責跟蹤當前的上下文session。Hibernate內建了此介面的兩種實現。

org.hibernate.context.JTASessionContext - 當前session根據JTA來跟蹤和界定。這和以前的僅支援JTA的方法是完全一樣的。詳情請參閱Javadoc。

org.hibernate.context.ThreadLocalSessionContext - 當前session通過當前執行的線程來跟蹤和界定。詳情也請參閱Javadoc。

這 兩種實現都提供了“每資料庫事務對應一個session”的編程模型,也稱作每次請求一個session。Hibernate session的起始和終結由資料庫事務的生存來控制。假若你採用自行編寫代碼來管理事務(比如,在純粹的J2SE,或者 JTA/UserTransaction/BMT),建議你使用Hibernate Transaction API來把底層事務實現從你的代碼中隱藏掉。如果你在支援CMT的EJB容器中執行,事務邊界是聲明式定義的,你不需要在代碼中進行任何事務或 session管理操作。請參閱第 11 章 事務和並發一節來閱讀更多的內容和範例程式碼。

hibernate.current_session_context_class 配置參數定義了應該採用哪個org.hibernate.context.CurrentSessionContext實現。注意,為了向下相容,如果未 配置此參數,但是存在org.hibernate.transaction.TransactionManagerLookup的配 置,Hibernate會採用org.hibernate.context.JTASessionContext。一般而言,此參數的值指明了要使用的實 現類的全名,但那兩個內建的實現可以使用簡寫,即"jta"和"thread"。

1、getCurrentSession()與openSession()的區別?

* 採用getCurrentSession()建立的session會綁定到當前線程中,而採用openSession() 
建立的session則不會 
* 採用getCurrentSession()建立的session在commit或rollback時會自動關閉,而採用openSession() 
建立的session必須手動關閉 
2、使用getCurrentSession()需要在hibernate.cfg.xml檔案中加入如下配置: 
* 如果使用的是本地事務(jdbc事務) 
<property name="hibernate.current_session_context_class">thread</property> 
* 如果使用的是全域事務(jta事務) 
<property name="hibernate.current_session_context_class">jta</property>

利於ThreadLocal模式管理Session 
   早在Java1.2推出之時,Java平台中就引入了一個新的支援:java.lang.ThreadLocal,給我們在編寫多線程程式 
   時提供了一種新的選擇。ThreadLocal是什麼呢?其實ThreadLocal並非是一個線程的本地實現版本,它並不是一個Thread, 
   而是thread local variable(線程局部變數)。也許把它命名為ThreadLocalVar更加合適。線程局部變數(ThreadLocal) 
   其實的功用非常簡單,就是為每一個使用某變數的線程都提供一個該變數值的副本,是每一個線程都可以獨立地改變自己的副本, 
   而不會和其它線程的副本衝突。從線程的角度看,就好像每一個線程都完全擁有一個該變數。 
   ThreadLocal是如何做到為每一個線程維護變數的副本的呢?其實實現的思路很簡單,在ThreadLocal類中有一個Map, 
   用於儲存每一個線程的變數的副本。比如下面的樣本實現(為了簡單,沒有考慮集合的泛型): 
public class HibernateUtil {

public static final ThreadLocal session =new ThreadLocal();

public static final SessionFactory sessionFactory; 
   static { 
      try { 
        sessionFactory = new Configuration().configure().buildSessionFactory(); 
      } catch (Throwable ex) { 
           throw new ExceptionInInitializerError(ex); 
      }     
}

     public static Session currentSession() throws HibernateException { 
        Session s = session.get(); 
        if(s == null) { 
          s = sessionFactory.openSession(); 
          session.set(s); 
           } 
         return s; 
       }

    public static void closeSession() throws HibernateException { 
           Session s = session.get(); 
        if(s != null) { 
            s.close(); 
        } 
        session.set(null); 
    } 
}

openSession() 與 getCurrentSession() 有何不同和關聯呢?

在 SessionFactory 啟動的時候, Hibernate 會根據配置建立相應的 CurrentSessionContext ,在 getCurrentSession() 被調用的時候,實際被執行的方法是 CurrentSessionContext.currentSession() 。在 currentSession() 執行時,如果當前 Session 為空白, currentSession 會調用 SessionFactory 的 openSession 。所以 getCurrentSession() 對於 Java EE 來說是更好的擷取 Session 的方法。

本文來自CSDN部落格,轉載請標明出處:http://blog.csdn.net/loveyout/archive/2009/05/17/4193894.aspx

hibernate 的SessionFactory的getCurrentSession 與 openSession() 的區別

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.