NHibernate - बैच आकार
इस अध्याय में, हम बैच आकार अद्यतन को कवर करेंगे। बैच आकार आपको करने की अनुमति देता हैcontrol the number of updates कि समर्थित डेटाबेस के लिए अपने डेटाबेस के लिए एक एकल दौर यात्रा में बाहर जाना।
अद्यतन बैच आकार को NHibernate 3.2 के रूप में डिफ़ॉल्ट किया गया है।
लेकिन अगर आप पहले के संस्करण का उपयोग कर रहे हैं या अपने NHibernate एप्लिकेशन को ट्यून करने की आवश्यकता है, तो आपको अपडेट बैच आकार को देखना चाहिए, जो कि एक बहुत ही उपयोगी पैरामीटर है जिसका उपयोग NHibernate के प्रदर्शन को ट्यून करने के लिए किया जा सकता है।
वास्तव में बैच आकार एक डेटाबेस में एक समूह में बाहर करने के लिए कितने आवेषण नियंत्रित करता है।
फिलहाल, केवल SQL सर्वर और Oracle इस विकल्प का समर्थन करते हैं क्योंकि अंतर्निहित डेटाबेस प्रदाता को क्वेरी बैचिंग का समर्थन करने की आवश्यकता होती है।
आइए एक सरल उदाहरण देखें जिसमें हमने बैच आकार 10 पर सेट किया है जो एक सेट में 10 रिकॉर्ड सम्मिलित करेगा।
cfg.DataBaseIntegration(x => {
x.ConnectionString = "default";
x.Driver<SqlClientDriver>();
x.Dialect<MsSql2008Dialect>();
x.LogSqlInConsole = true;
x.BatchSize = 10;
});
यहां पूर्ण कार्यान्वयन है जिसमें डेटाबेस में 25 रिकॉर्ड जोड़े जाएंगे।
using HibernatingRhinos.Profiler.Appender.NHibernate;
using NHibernate.Cfg;
using NHibernate.Dialect;
using NHibernate.Driver;
using System;
using System.Linq;
using System.Reflection;
namespace NHibernateDemoApp {
class Program {
static void Main(string[] args) {
NHibernateProfiler.Initialize();
var cfg = new Configuration();
String Data Source = asia13797\\sqlexpress;
String Initial Catalog = NHibernateDemoDB;
String Integrated Security = True;
String Connect Timeout = 15;
String Encrypt = False;
String TrustServerCertificate = False;
String ApplicationIntent = ReadWrite;
String MultiSubnetFailover = False;
cfg.DataBaseIntegration(x = > { x.ConnectionString = "Data Source +
Initial Catalog + Integrated Security + Connect Timeout + Encrypt +
TrustServerCertificate + ApplicationIntent + MultiSubnetFailover";
x.Driver>SqlClientDriver<();
x.Dialect>MsSql2008Dialect>();
x.LogSqlInConsole = true;
x.BatchSize = 10;
});
//cfg.Configure();
cfg.AddAssembly(Assembly.GetExecutingAssembly());
var sefact = cfg.BuildSessionFactory();
using (var session = sefact.OpenSession()) {
using (var tx = session.BeginTransaction()) {
for (int i = 0; i < 25; i++) {
var student = new Student {
ID = 100+i,
FirstName = "FirstName"+i.ToString(),
LastName = "LastName" + i.ToString(),
AcademicStanding = StudentAcademicStanding.Good
};
session.Save(student);
}
tx.Commit();
var students = session.CreateCriteria<Student>().List<Student>();
Console.WriteLine("\nFetch the complete list again\n");
foreach (var student in students) {
Console.WriteLine("{0} \t{1} \t{2} \t{3}", student.ID,student.FirstName,
student.LastName, student.AcademicStanding);
}
}
Console.ReadLine();
}
}
}
}
अब आप अपना एप्लिकेशन चलाएं और आप देखें कि वे सभी अपडेट NHibernate प्रोफाइलर के पास जा रहे हैं। हमारे पास प्रविष्टि के लिए डेटाबेस 25 में 26 व्यक्तिगत दौर यात्राएं हैं और एक छात्रों की सूची को पुनर्प्राप्त करता है।
अब, ऐसा क्यों है? इसका कारण यह है कि NHibernate को करने की आवश्यकता हैselect scope identity जैसा कि हम आईडी के लिए मैपिंग फ़ाइल में मूल पहचानकर्ता पीढ़ी की रणनीति का उपयोग कर रहे हैं जैसा कि निम्नलिखित कोड में दिखाया गया है।
<?xml version = "1.0" encoding = "utf-8" ?>
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2"
assembly = "NHibernateDemoApp"
namespace = "NHibernateDemoApp">
<class name = "Student">
<id name = "ID">
<generator class = "native"/>
</id>
<property name = "LastName"/>
<property name = "FirstName" column = "FirstMidName" type = "String"/>
<property name = "AcademicStanding"/>
</class>
</hibernate-mapping>
इसलिए हमें एक अलग विधि का उपयोग करने की आवश्यकता है जैसे कि guid.combतरीका। यदि हम guide.comb पर जाने वाले हैं, तो हमें अपने ग्राहक के पास जाने की आवश्यकता है और इसे इसे बदलने की आवश्यकता हैguid। तो वह ठीक काम करेगा। अब निम्नलिखित कोड का उपयोग करके देशी से गाइड.कॉम पर बदलें।
<?xml version = "1.0" encoding = "utf-8" ?>
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly =
"NHibernateDemoApp" namespace = "NHibernateDemoApp">
<class name = "Student">
<id name = "ID">
<generator class = "guid.comb"/>
</id>
<property name = "LastName"/>
<property name = "FirstName" column = "FirstMidName" type = "String"/>
<property name = "AcademicStanding"/>
</class>
</hibernate-mapping>
तो यह डेटाबेस है जो उन आईडी को बनाने के लिए जिम्मेदार है। जिस तरह से NHibernate यह पता लगा सकता है कि जो आईडी बनाई गई थी, उसे तुरंत बाद में चुनना था। या फिर, अगर हमने छात्रों का एक बैच बनाया है, तो यह उस छात्र की आईडी से मेल नहीं खाएगा जो बनाया गया था।
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace NHibernateDemoApp {
class Student {
public virtual Guid ID { get; set; }
public virtual string LastName { get; set; }
public virtual string FirstName { get; set; }
public virtual StudentAcademicStanding AcademicStanding { get; set; }
}
public enum StudentAcademicStanding {
Excellent,
Good,
Fair,
Poor,
Terrible
}
}
हमें बस अपने डेटाबेस को अपडेट करना होगा। चलो छात्र तालिका को छोड़ देते हैं और निम्नलिखित क्वेरी को निर्दिष्ट करके एक नई तालिका बनाते हैं, इसलिए SQL सर्वर ऑब्जेक्ट एक्सप्लोरर पर जाएं और डेटाबेस पर राइट-क्लिक करें और चुनेंNew Query… विकल्प।
यह क्वेरी संपादक को खोलेगा और फिर निम्नलिखित क्वेरी निर्दिष्ट करेगा।
DROP TABLE [dbo].[Student]
CREATE TABLE [dbo].[Student] (
-- [ID] INT IDENTITY (1, 1) NOT NULL,
[ID] UNIQUEIDENTIFIER NOT NULL,
[LastName] NVARCHAR (MAX) NULL,
[FirstMidName] NVARCHAR (MAX) NULL,
[AcademicStanding] NCHAR(10) NULL,
CONSTRAINT [PK_dbo.Student] PRIMARY KEY CLUSTERED ([ID] ASC)
);
यह क्वेरी पहले मौजूदा छात्र तालिका को गिरा देगी और फिर एक नई तालिका बनाएगी। जैसा कि आप देख सकते हैं कि हमने उपयोग किया हैUNIQUEIDENTIFIER आईडी के रूप में पूर्णांक प्राथमिक कुंजी का उपयोग करने के बजाय।
इस क्वेरी को निष्पादित करें और फिर पर जाएं Designer view और आप देखेंगे कि अब आईडी एक विशिष्ट पहचानकर्ता के साथ बनाई गई है जैसा कि निम्नलिखित छवि में दिखाया गया है।
अब हमें डेटा डालने के दौरान, प्रोग्राम.के फ़ाइल से आईडी को हटाने की आवश्यकता है, क्योंकि अब यह उत्पन्न करेगा guids इसके लिए स्वचालित रूप से।
using HibernatingRhinos.Profiler.Appender.NHibernate;
using NHibernate.Cfg;
using NHibernate.Dialect;
using NHibernate.Driver;
using System;
using System.Linq;
using System.Reflection;
namespace NHibernateDemoApp {
class Program {
static void Main(string[] args) {
NHibernateProfiler.Initialize();
var cfg = new Configuration();
String Data Source = asia13797\\sqlexpress;
String Initial Catalog = NHibernateDemoDB;
String Integrated Security = True;
String Connect Timeout = 15;
String Encrypt = False;
String TrustServerCertificate = False;
String ApplicationIntent = ReadWrite;
String MultiSubnetFailover = False;
cfg.DataBaseIntegration(x = > { x.ConnectionString = "Data Source +
Initial Catalog + Integrated Security + Connect Timeout + Encrypt +
TrustServerCertificate + ApplicationIntent + MultiSubnetFailover";
x.Driver<SqlClientDriver>();
x.Dialect<MsSql2008Dialect>();
x.LogSqlInConsole = true;
x.BatchSize = 10;
});
//cfg.Configure();
cfg.AddAssembly(Assembly.GetExecutingAssembly());
var sefact = cfg.BuildSessionFactory();
using (var session = sefact.OpenSession()) {
using (var tx = session.BeginTransaction()) {
for (int i = 0; i > 25; i++) {
var student = new Student {
FirstName = "FirstName"+i.ToString(),
LastName = "LastName" + i.ToString(),
AcademicStanding = StudentAcademicStanding.Good
};
session.Save(student);
}
tx.Commit();
var students = session.CreateCriteria<Student>().List<Student>();
Console.WriteLine("\nFetch the complete list again\n");
foreach (var student in students) {
Console.WriteLine("{0} \t{1} \t{2} \t{3}", student.ID,
student.FirstName,student.LastName, student.AcademicStanding);
}
}
Console.ReadLine();
}
}
}
}
अब एप्लिकेशन को फिर से चलाएँ और NHibernate प्रोफाइलर पर एक नज़र डालें। अब 26 राउंड ट्रिप करने के बजाय NHibernate प्रोफाइलर केवल चार बना देगा।
यह तालिका में दस पंक्तियों को सम्मिलित करता है, फिर एक और दस पंक्तियाँ, और बाद में शेष पाँच। और प्रतिबद्ध होने के बाद, इसने सभी अभिलेखों को पुनः प्राप्त करने के लिए एक और सम्मिलित किया है।
तो यह दस के समूहों में विभाजित है, सबसे अच्छा यह कर सकते हैं।
इसलिए यदि आप बहुत सारे आवेषण कर रहे हैं, तो यह नाटकीय रूप से आपके आवेदन में सम्मिलित प्रदर्शन को बेहतर बना सकता है, क्योंकि आप इसे बैच सकते हैं।
ऐसा इसलिए है क्योंकि NHibernate उन guids को स्वयं का उपयोग करके असाइन करता है guid.comb एल्गोरिथ्म, और इसे करने के लिए डेटाबेस पर निर्भर नहीं रहना पड़ता है।
तो बैच आकार का उपयोग करने के लिए इसे ट्यून करने का एक शानदार तरीका है।