標籤:style blog class code java tar
NHibernate的基本特徵是完成物件導向的程式設計語言到關聯式資料庫的映射,在NHibernate中使用持久化對象PO(Persistent Object)完成持久化操作,對PO的操作必須在Session管理下才能同步到資料庫, 但是這裡的Session並非指HttpSession,可以理解為基於ADO.NET的Connnection,Session是NHibernate運作的中心,對象的生命週期、事務的管理、資料庫的存取都與Session息息相關,首先,我們需要知道, SessionFactory負責建立Session,SessionFactory是安全執行緒的,多個並發線程可以同時訪問一個SessionFactory 並從中擷取Session執行個體。 只有一個資料存放區源(資料庫),只需建立一個SessionFactory,SessionFactory就是個重量級對象,如果應用只有一個資料存放區源,只需建立一個SessionFactory執行個體,因為隨意地建立SessionFactory執行個體會佔用大量記憶體空間。這裡所說的只建立一個,通常是使用單例模式,使本次應用程式定義域中的所有請求都訪問同一個SessionFactory.但是他的執行個體session是輕量級的。而Session並非安全執行緒,也就是說,如果多個線程同時使用一個Session執行個體進行資料存取,則將會導致Session 資料存取邏輯混亂.因此建立的Session執行個體必須在本地存取空上運行,使之總與當前的線程相關。而Session並非安全執行緒,也就是說,如果多個線程同時使用一個Session執行個體進行資料存取,則將會導致 Session 資料存取邏輯混亂.因此建立的Session執行個體必須在本地存取空上運行, 使之總與當前的線程相關。
多個線程中(即多個request中)不能使用一個Session,但一個request中可能存在多個Session,不從效能角度考慮,這樣也是不會報錯的,就像ADO.NET的時候,在一個方法中同時建立了兩個Connection來操作同一個資料庫一樣,雖然效能有問題,但是還是能實現操作的。注意,如果涉及到了Transaction交易處理,那就必須在同一個Session中完成。
為了保證一個瀏覽器訪問,也就是一次request,使用的都是各自的Session,在Java(Hibernate)中,ThreadLocal 是Java中一種較為特殊的線程綁定機制,通過ThreadLocal存取的資料, 總是與當前線程相關, 也就是說,JVM 為每個啟動並執行線程,綁定了私人的本地執行個體存取空間,從而為多線程環境常出現的並發訪問問題提供了一種隔離機,ThreadLocal並不是執行緒區域化的實現,而是線程局部變數。也就是說每個使用該變數的線程都必須為該變數提供一個副本,每個線程改變該變數的值僅僅是改變該副本的值,而不會影響其他線程的該變數的值,ThreadLocal是隔離多個線程的資料共用,不存在多個線程之間共用資源,因此不再需要對線程同步。
ASP.NET MVC中,通常會結合Controller使用。一般是在controller中調用各個domain的service方法最終進行了資料庫的各種操作,所以我們只要在controller的action執行之前,把一個新的session綁定到當前的線程就可以了,本次request流程中,所有對於資料庫的操作,都使用這個Session.
AttributeUsageAttribute(AttributeTargets.Method, Inherited = false, AllowMultiple = false)] public class NHibernateSessionAttribute : FilterAttribute, IActionFilter { public NHibernateSessionAttribute() : base() { } public void OnActionExecuting(ActionExecutingContext filterContext) { var session = SessionProvider.GetNewSession();//這裡是擷取了一個新的Session.因為這裡是每次操作剛剛開始的設計NHibernate CurrentSessionContext.Bind(session);//開啟一個新的Session,放入到當前上下文中,以後再用Session的時候GetCurrentSession就可以 } public void OnActionExecuted(ActionExecutedContext filterContext) { CurrentSessionContext.Unbind(SessionProvider.GetSessionFactory()).Close(); }在使用的時候,只需要在Action上邊加一個NHibernateSession的Filter,就可以在這個Action執行前,先執行這個Filter的內容,綁定Session,然後完事後,在解除Session
[NHibernateSession] public ActionResult SaveNewOrder(Order order) { ...... return RedirectToAction("Index"); }
SessionProvider使用者管理Session的擷取,是一個靜態類,保證了SessionFactory唯一性,這樣做比較考慮效能。
public class SessionProvider { private static ISessionFactory sessionFactory; public static NHibernate.Cfg.Configuration configuration; static SessionProvider() { configuration = new NHibernate.Cfg.Configuration().Configure(); sessionFactory = configuration.BuildSessionFactory(); } public static ISessionFactory GetSessionFactory() { return sessionFactory; } public static ISession GetNewSession() { return sessionFactory.OpenSession(); //這個方法一般是首次用到Session的時候,然後綁定到上下文中 } public static ISession GetNewOrCurrentSession() { ISession session = null; if (!CurrentSessionContext.HasBind(sessionFactory)) { session = sessionFactory.OpenSession(); CurrentSessionContext.Bind(session); } else { session = sessionFactory.GetCurrentSession(); if (!session.IsOpen) session = sessionFactory.OpenSession(); } return session; } }
注意:如果我們在建立Windwos Service的時候,就沒有了Action中的那個綁定以及解除綁定的過程。這裡的Session是Thread_Static的,所以一用到Session的時候,就會直接到Session = SessionProvider.GetNewOrCurrentSession中來擷取。但是注意,這裡並沒有一個解除綁定的過程。在ASP.NET MVC的是,unbind的時候,會自動調用Flush方法,然後去更新資料庫。而如果我們在Windows Service程式中不手動調用Session.Flush去更新資料庫,一切操作是沒有同步到資料庫中的。當這個線程結束,Session消失,一切操作都沒有同步。所以,我們要記得手動去調用Flush方法。