久久99热66热这里只有精品,特黄特色的大片在线观看,亚洲日本三级在线观看,国产三级农村妇女在线,亚洲av毛片免费在线观看,哺乳叫自慰在线看,天天干美女av网

[NHibernate]立即加載 -電腦資料

電腦資料 時間:2019-01-01 我要投稿
【m.dameics.com - 電腦資料】

    一個例子

    1、一對多關(guān)系

    通過立即加載的方式,加載客戶信息及客戶的訂單信息,

[NHibernate]立即加載

。

    復(fù)制代碼

    1    ///

    2    /// 采用立即加載的方式加載客戶信息

    3    ///

    4    ///

    5    ///

    6    public Customer GetCustomerByImmediatelyLoad(Guid customerID)

    7    {

    8      //獲得ISession實例

    9      //通過using的方式將session釋放,為了保證是立即加載的數(shù)據(jù),而不是延遲加載的

    10      //還記得上篇文章中提到的在延遲加載中,在關(guān)閉session情況下,再讀取數(shù)據(jù)的時候就會有一個異常信息

    11      //忘記的可以再回到上一篇文章進(jìn)行查看

    12      using (ISession session = NHibernateHelper.GetSession())

    13      {

    14        return session.Get(customerID);

    15      }

    16    }

    復(fù)制代碼

    1.1、修改Customer.hbm.xml文件

    因為nhibernate默認(rèn)是使用Lazy的方式加載數(shù)據(jù)的,也就是默認(rèn)Lazy=“true”的,需要顯示的指定Lazy=“false”。

    1  

    2  

    3  

    4   

    5   

    6  

    編寫一個測試驗證,調(diào)用數(shù)據(jù)訪問層中的使用using強(qiáng)制資源清理Session加載Customer對象的方法加載一個Customer對象,NHibernate這時立即加載Customer相關(guān)聯(lián)的Order對象。

    利用NHibernate提供實用類(NHibernateUtil)測試被關(guān)聯(lián)的Customer對象集合是否已初始化(也就是已加載)。

    上篇文章中也說明了,如果在關(guān)閉session的情況下,如果你展開customer的orders屬性,就會出現(xiàn)異常

    這也說明了Orders是跟customer一樣立即加載的。

    查看生成的sql語句

    1 exec sp_executesql N'SELECT customer0_.CustomerID as CustomerID0_0_, customer0_.Version as Version0_0_, customer0_.CustomerName as Customer3_0_0_, customer0_.CustomerAddress as Customer4_0_0_ FROM TB_Customer customer0_ WHERE customer0_.CustomerID=@p0',N'@p0 uniqueidentifier',@p0='B0720295-9541-40B3-9994-610066224DB8'

    2

    3 exec sp_executesql N'SELECT orders0_.CustomerID as CustomerID1_, orders0_.OrderID as OrderID1_, orders0_.OrderID as OrderID1_0_, orders0_.OrderDate as OrderDate1_0_, orders0_.CustomerID as CustomerID1_0_ FROM TB_Order orders0_ WHERE orders0_.CustomerID=@p0',N'@p0 uniqueidentifier',@p0='B0720295-9541-40B3-9994-610066224DB8'

    第一條sql為查詢Customer的,第二條sql是查詢與該customer相關(guān)聯(lián)的order的。

    1.2、使用NHibernateUtil實用類

    NHibernate提供實用類(NHibernateUtil)可以用來檢測被關(guān)聯(lián)的對象集合是否已初始化,還可以強(qiáng)制初始化未初始化的相關(guān)聯(lián)的對象。有了這個功能,我們就可以修改數(shù)據(jù)訪問層中的方法,把上面使用Using強(qiáng)制清理關(guān)閉Session的方法中加上NHibernateUtil類提供Initialize方法來初始化Customer相關(guān)聯(lián)的Order對象集合。

    修改Customer.hbm.xml文件,將set節(jié)點的lazy屬性設(shè)置為默認(rèn)值true,或者刪除該屬性。

    復(fù)制代碼

    1    ///

    2    /// NHibernateUtil方式,立即加載客戶信息及關(guān)聯(lián)的數(shù)據(jù)

    3    ///

    4    ///

    5    ///

    6    public Customer GetCustomerByImmediatelyLoadNHibernateUtil(Guid customerID)

    7    {

    8      //獲得ISession實例

    9      //通過using的方式將session釋放,為了保證是立即加載的數(shù)據(jù),而不是延遲加載的

    10      //還記得上篇文章中提到的在延遲加載中,在關(guān)閉session情況下,再讀取數(shù)據(jù)的時候就會有一個異常信息

    11      //忘記的可以再回到上一篇文章進(jìn)行查看

    12      using (ISession session = NHibernateHelper.GetSession())

    13      {

    14        Customer customer = session.Get(customerID);

    15        //

    16        // 摘要:

    17        //  Force initialization of a proxy or persistent collection.

    18        //

    19        NHibernate.NHibernateUtil.Initialize(customer.Orders);

    20        return customer;

    21      }

    22    }

    復(fù)制代碼

    通過該種方式,跟上面的測試結(jié)果相同,就不再貼圖了,

電腦資料

[NHibernate]立即加載》(http://m.dameics.com)。

    2、多對多關(guān)系

    2.1、使用Lazy=“false”屬性

    例子同上面的類似,就不再贅述。

    2.2、使用NHibernateUtil實用類

    這里采用Order和Product多對多實例進(jìn)行分析。

    如果想在加載Order的情況下,加載該Order下的所有Product,可以使用NHibernateUtil類初始化關(guān)聯(lián)對象(把他們從數(shù)據(jù)庫取出來)。

    復(fù)制代碼

    1    ///

    2    /// NHibernateUtil方式,立即加載訂單與商品

    3    ///

    4    ///

    5    ///

    6    public Order GetOrderProductByImmediatelyLoadNHibernateUtil(Guid orderId)

    7    {

    8      using (var session = NHibernateHelper.GetSession())

    9      {

    10        var rder = session.Get(orderId);

    11        //強(qiáng)制初始化customer

    12        NHibernate.NHibernateUtil.Initialize(order.Customer);

    13        //強(qiáng)制初始化Product

    14        NHibernate.NHibernateUtil.Initialize(order.Products);

    15        return order;

    16      }

    17    }

    1 exec sp_executesql N'SELECT order0_.OrderID as OrderID1_0_, order0_.OrderDate as OrderDate1_0_, order0_.CustomerID as CustomerID1_0_ FROM TB_Order order0_ WHERE order0_.OrderID=@p0',N'@p0 uniqueidentifier',@p0='78A53F67-A293-48A1-BBE2-86FED77342FA'

    2

    3 exec sp_executesql N'SELECT customer0_.CustomerID as CustomerID0_0_, customer0_.Version as Version0_0_, customer0_.CustomerName as Customer3_0_0_, customer0_.CustomerAddress as Customer4_0_0_ FROM TB_Customer customer0_ WHERE customer0_.CustomerID=@p0',N'@p0 uniqueidentifier',@p0='B0720295-9541-40B3-9994-610066224DB8'

    4

    5 exec sp_executesql N'SELECT products0_.OrderID as OrderID1_, products0_.ProductID as ProductID1_, product1_.ProductID as ProductID3_0_, product1_.Name as Name3_0_, product1_.Price as Price3_0_ FROM TB_OrderProduct products0_ left outer join TB_Product product1_ on products0_.ProductID=product1_.ProductID WHERE products0_.OrderID=@p0',N'@p0 uniqueidentifier',@p0='78A53F67-A293-48A1-BBE2-86FED77342FA'

    復(fù)制代碼

    這里生成了三條sql語句,查詢和Order關(guān)聯(lián)的customer和Product。

    2.3使用HQL抓取策略

    HQL語句支持的連接類型為:inner join(內(nèi)連接)、left outer join(左外連接)、right outer join(右外連接)、full join(全連接,不常用)。

    “抓取fetch”連接允許僅僅使用一個選擇語句就將相關(guān)聯(lián)的對象隨著他們的父對象的初始化而被初始化,可以有效的代替了映射文件中的外聯(lián)接與延遲屬性聲明。

    幾點注意:

    fetch不與setMaxResults() 或setFirstResult()共用,因為這些操作是基于結(jié)果集的,而在預(yù)先抓取集合時可能包含重復(fù)的數(shù)據(jù),也就是說無法預(yù)先知道精確的行數(shù)。

    fetch還不能與獨立的with條件一起使用。通過在一次查詢中fetch多個集合,可以制造出笛卡爾積,因此請多加注意。對多對多映射來說,同時join fetch多個集合角色可能在某些情況下給出并非預(yù)期的結(jié)果,也請小心。

    使用full join fetch 與 right join fetch是沒有意義的。 如果你使用屬性級別的延遲獲取,在第一個查詢中可以使用 fetch all properties 來強(qiáng)制NHibernate立即取得那些原本需要延遲加載的屬性。

    參考:http://www.cnblogs.com/lyj/archive/2008/10/29/1322373.html

    測試代碼

    復(fù)制代碼

    1    ///

    2    /// HQL方式,立即加載訂單與商品

    3    ///

    4    ///

    5    ///

    6    public Order GetOrderProductByImmediatelyLoadHQL(Guid orderId)

    7    {

    8      using (var session = NHibernateHelper.GetSession())

    9      {

    10        return session.CreateQuery("from Order o" +

    11          " left outer join fetch o.Products" +

    12          " inner join fetch o.Customer where o.OrderID=:orderId")

    13          .SetGuid("orderId", orderId)

    14          .UniqueResult();

    15      }

    16    }

    上篇文章中,描述了一個N+1的問題,這里再次說明一下,N+1的問題,我是這樣理解的你本來想要一條,可是采用默認(rèn)延遲加載的情況下,就會把所有的信息都加載,有點浪費,而這時采用立即加載的方式,你可以很好的控制,想加載幾條加載幾條。

    總結(jié)

    這篇文章介紹了nhibernate立即加載的方式,什么時候使用立即加載,什么時候使用延遲加載,視項目中的情況而定。比如這種情況,如果在多對多的關(guān)系中,你想加載幾條數(shù)據(jù),也就是N+1的問題,還是采用立即加載的方式更好,此時,不會加載多余的數(shù)據(jù),可以更好的達(dá)到控制的目的。

最新文章