NHibernate - आलसी लोड हो रहा है
इस अध्याय में, हम आलसी लोडिंग सुविधा को कवर करेंगे। यह डिफ़ॉल्ट रूप से एक पूरी तरह से अलग अवधारणा है और NHibernate में आलसी लोडिंग नहीं है, उदाहरण के लिए यदि आप एक ग्राहक को लोड करते हैं, तो यह सभी ऑर्डर लोड नहीं करने वाला है।
आदेश संग्रह मांग पर लोड किया जाएगा।
कोई भी एसोसिएशन, चाहे वह कई-से-एक हो या एक संग्रह डिफ़ॉल्ट रूप से आलसी हो, उसे एक की आवश्यकता होती है Open ISession।
यदि आपने अपना सत्र बंद कर दिया है, या यदि आपने अपना लेन-देन किया है, तो आप एक आलसी लोड अपवाद प्राप्त कर सकते हैं, जो उन अतिरिक्त वस्तुओं में खींच नहीं सकता है।
आपको आलसी लोडिंग के बारे में सावधान रहना होगा और आपको वास्तव में कितने डेटा की आवश्यकता होगी।
आप पूरे संघ के लिए आलसी लोडिंग को बंद कर सकते हैं या आप आलसी को झूठे बराबर कर सकते हैं या आप एक लचर रणनीति भी निर्दिष्ट कर सकते हैं।
यहाँ है Program.cs फ़ाइल कार्यान्वयन।
using System;
using System.Data;
using System.Linq;
using System.Reflection;
using HibernatingRhinos.Profiler.Appender.NHibernate;
using NHibernate.Cfg;
using NHibernate.Dialect;
using NHibernate.Driver;
using NHibernate.Linq;
namespace NHibernateDemo {
internal class Program {
private static void Main() {
var cfg = ConfigureNHibernate();
var sessionFactory = cfg.BuildSessionFactory();
Guid id;
using(var session = sessionFactory.OpenSession())
using(var tx = session.BeginTransaction()) {
var newCustomer = CreateCustomer();
Console.WriteLine("New Customer:");
Console.WriteLine(newCustomer);
session.Save(newCustomer);
id = newCustomer.Id;
tx.Commit();
}
using(var session = sessionFactory.OpenSession())
using(var tx = session.BeginTransaction()) {
var reloaded = session.Load<Customer>(id);
Console.WriteLine("Reloaded:");
Console.WriteLine(reloaded);
Console.WriteLine("The orders were ordered by: ");
foreach (var order in reloaded.Orders) {
Console.WriteLine(order.Customer);
}
tx.Commit();
}
Console.WriteLine("Press <ENTER> to exit...");
Console.ReadLine();
}
private static Customer CreateCustomer() {
var customer = new Customer {
FirstName = "John",
LastName = "Doe",
Points =100,
HasGoldStatus = true,
MemberSince = new DateTime(2012, 1, 1),
CreditRating = CustomerCreditRating.Good,
AverageRating = 42.42424242,
Address = CreateLocation()
};
var order1 = new Order { Ordered = DateTime.Now };
customer.AddOrder(order1);
var order2 = new Order {
Ordered = DateTime.Now.AddDays(-1),
Shipped = DateTime.Now,
ShipTo = CreateLocation()
};
customer.AddOrder(order2); return customer;
}
private static Location CreateLocation() {
return new Location {
Street = "123 Somewhere Avenue",
City = "Nowhere",
Province = "Alberta",
Country = "Canada"
};
}
private static Configuration ConfigureNHibernate() {
NHibernateProfiler.Initialize();
var cfg = new Configuration();
cfg.DataBaseIntegration(x => {
x.ConnectionStringName = "default";
x.Driver<SqlClientDriver>();
x.Dialect<MsSql2008Dialect<();
x.IsolationLevel = IsolationLevel.RepeatableRead;
x.Timeout = 10;
x.BatchSize = 10;
});
cfg.SessionFactory().GenerateStatistics();
cfg.AddAssembly(Assembly.GetExecutingAssembly());
return cfg;
}
}
}
इसे समझने के लिए, आइए एप्लिकेशन को चलाएं और NHibernate Profiler पर एक नज़र डालें।
जैसा कि आप देख सकते हैं कि हमारे पास सेलेक्ट फ्रॉम कस्टमर है, जिसे एक विशेष कस्टमर आईडी दी गई है और फिर हमारे पास एक और सिलेक्ट फ्रॉम ऑर्डर्स टेबल है, जब वह वास्तव में उस ग्राहक के संग्रह तक पहुंचता है।
इसलिए हमारे पास डेटाबेस के लिए 2 राउंडट्रिप्स हैं। अब, कभी-कभी, हम इसे अनुकूलित करना चाहेंगे। ऐसा करने के लिए, आइए चलते हैंcustomer.hbm.xml फ़ाइल करें और एक फ़ेचिंग रणनीति जोड़ें और इसे एक सम्मिलित रूप से करने के लिए कहें।
<?xml version = "1.0" encoding = "utf-8" ?>
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly = "NHibernateDemo"
namespace = "NHibernateDemo">
<class name = "Customer">
<id name = "Id">
<generator class = "guid.comb"/>
</id>
<property name = "FirstName"/>
<property name = "LastName"/>
<property name = "AverageRating"/>
<property name = "Points"/>
<property name = "HasGoldStatus"/>
<property name = "MemberSince" type = "UtcDateTime"/>
<property name = "CreditRating" type = "CustomerCreditRatingType"/>
<component name = "Address">
<property name = "Street"/>
<property name = "City"/>
<property name = "Province"/>
<property name = "Country"/>
</component>
<set name = "Orders" table = "`Order`" cascade = "all-delete-orphan"
fetch = "join">
<key column = "CustomerId"/>
<one-to-many class = "Order"/>
</set>
</class>
</hibernate-mapping>
जैसा कि आप देख सकते हैं कि हमने अपने आवेदन में कोई कोड नहीं बदला है, हमने अभी-अभी इसमें एक शामिल करने की रणनीति जोड़ी है customer.hbm.xml। चलो इस एप्लिकेशन को फिर से चलाते हैं, यह अभी भी उसी तरह व्यवहार करता है। आइए NHibernate Profiler को देखें।
इससे पहले, प्रोग्राम में डेटाबेस के लिए दो राउंड ट्रिप थे, अब, इसमें केवल एक है और ऐसा इसलिए है क्योंकि यह एक लेफ्ट बाहरी जॉइन कर रहा है।
हम यह देख सकते हैं कि यह ग्राहक आईडी और ग्राहक आईडी के आधार पर ऑर्डर टेबल के बीच एक बाएं बाहरी जुड़ाव कर रहा है, और इसलिए, यह एक ही बार में सभी जानकारी लोड करने में सक्षम है।
हमने डेटाबेस के लिए 1 राउंडट्रिप को बचाया है।
नीचे की ओर यह है कि ग्राहक जानकारी को दोनों लाइनों पर डुप्लिकेट किया जाएगा और इस तरह से कि एक एसक्यूएल बाहरी बाहरी कार्यों में शामिल होता है।
इसलिए, इस रणनीति के साथ, हम थोड़ा और डेटा वापस खींच रहे हैं और हम एक राउंडट्रिप को बचा रहे हैं।
आप इसे क्वेरी स्तर पर भी कर सकते हैं, तो चलिए चलते हैं Program.cs फ़ाइल और सरल पुनः लोड किए गए उदाहरण को देखें।
using(var session = sessionFactory.OpenSession())
using(var tx = session.BeginTransaction()) {
//var query = from customer in session.Query<Customer>()
// select customer;
//var reloaded = query.Fetch(x => x.Orders).ToList();
var reloaded = session.Load<Customer>(id);
Console.WriteLine("Reloaded:");
Console.WriteLine(reloaded);
Console.WriteLine("The orders were ordered by: ");
foreach (var order in reloaded.Orders) {
Console.WriteLine(order.Customer);
}
tx.Commit();
}
यहां, हम ग्राहक द्वारा एक लोड कर रहे हैं। अब इसे एक क्वेरी में बदलते हैं और हम निम्न लिंक में दिखाए अनुसार लिंक क्वेरी का उपयोग करेंगे।
using(var session = sessionFactory.OpenSession())
using(var tx = session.BeginTransaction()) {
var query = from customer in session.Query<Customer>()
where customer.Id == id select customer;
var reloaded = query.Fetch(x => x.Orders).ToList().First();
Console.WriteLine("Reloaded:");
Console.WriteLine(reloaded);
tx.Commit();
}
आइए, हम भी इसमें से रणनीति निकालते हैं customer.hbm.xml फ़ाइल।
<?xml version = "1.0" encoding = "utf-8" ?>
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly = "NHibernateDemo"
namespace = "NHibernateDemo">
<class name = "Customer">
<id name = "Id">
<generator class = "guid.comb"/>
</id>
<property name = "FirstName"/>
<property name = "LastName"/>
<property name = "AverageRating"/>
<property name = "Points"/>
<property name = "HasGoldStatus"/>
<property name = "MemberSince" type = "UtcDateTime"/>
<property name = "CreditRating" type = "CustomerCreditRatingType"/>
<component name = "Address">
<property name = "Street"/>
<property name = "City"/>
<property name = "Province"/>
<property name = "Country"/>
</component>
<set name = "Orders" table = "`Order`" cascade = "all-delete-orphan">
<key column = "CustomerId"/>
<one-to-many class = "Order"/>
</set>
</class>
</hibernate-mapping>
चलिए इस एप्लिकेशन को फिर से चलाते हैं और आपको निम्न आउटपुट दिखाई देगा।
New Customer:
John Doe (00000000-0000-0000-0000-000000000000)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Unspecified)
CreditRating: Good
AverageRating: 42.42424242
Orders:
Order Id: 00000000-0000-0000-0000-000000000000
Order Id: 00000000-0000-0000-0000-000000000000
Reloaded:
John Doe (6ebacd17-f9ba-4ad8-9817-a5bb01112a5a)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: 16a6596b-d56e-41c7-9681-a5bb01112a60
Order Id: d41d615b-0f21-4032-81db-a5bb01112a61
Press <ENTER> to exit...
अब आइए NHibernate Profiler को देखें, आप देख सकते हैं कि हमें यह उत्सुकता एक बार फिर से हो रही है, लेकिन इस बार, यह क्वेरी पर आधारित है।