NHibernate - QueryOver क्वेरी

इस अध्याय में, हम QueryOver Queries को कवर करेंगे। यह एक नया सिंटैक्स है जो विधि श्रृंखला सिंटैक्स का उपयोग करके LINQ की तरह अधिक है जैसा कि निम्नलिखित क्वेरी में दिखाया गया है।

var customers = session.QueryOver<Customer>() .Where(x => x.FirstName == "Laverne");
  • यह अभी भी कवर के तहत मानदंड है, लेकिन अब हमारे प्रश्न दृढ़ता से टाइप किए गए हैं।

  • जैसा कि हमने मापदंड क्वेरी में देखा है, पहला नाम सिर्फ एक अपारदर्शी स्ट्रिंग है, अब हम वास्तव में एक का उपयोग कर रहे हैं x.FirstName, इसलिए पहला नाम रिफैक्ट हो जाता है और नाम बदल दिया जाता है जो लिंक शैली मापदंड क्वेरी में क्वेरी का उपयोग करके बदल जाता है।

  • हम अभी भी कई समान काम कर सकते हैं, लेकिन आप क्वेरी को समझने के साथ क्वेरी कॉम्प्रिहेंशन सिंटैक्स का उपयोग नहीं कर सकते हैं, आपको विधि श्रृंखला सिंटैक्स का उपयोग करना होगा और आप लिंक और मापदंड को मिक्स और मैच नहीं कर सकते।

  • बहुत सारे प्रश्नों के लिए, एपीआई पर क्वेरी बहुत उपयोगी है और सीधे क्राइटेरिया का उपयोग करने की तुलना में ऑब्जेक्ट सिंटैक्स को समझना बहुत आसान है।

आइए एक सरल उदाहरण देखें, जिसमें हम एक ग्राहक को पुनः प्राप्त करेंगे, जिसका पहला नाम Laverne एक क्वेरी ओवर का उपयोग करता है।

using System; 
using System.Data; 
using System.Linq; 
using System.Reflection; 

using HibernatingRhinos.Profiler.Appender.NHibernate; 
using NHibernate.Cfg; 
using NHibernate.Criterion; 
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();
         using(var session = sessionFactory.OpenSession()) 
         
         using(var tx = session.BeginTransaction()) { 
            var customers = session.QueryOver<Customer>() 
               .Where(x => x.FirstName == "Laverne"); 
            
            foreach (var customer in customers.List()) { 
               Console.WriteLine(customer); 
            } 
				
            tx.Commit(); 
         }
			
         Console.WriteLine("Press <ENTER> to exit..."); 
         Console.ReadLine(); 
      }
      
      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; 
      } 
   } 
}

जैसा कि आप देख सकते हैं कि यह अभी भी कवर के नीचे मानदंड है, लेकिन सिर्फ एक अच्छा वाक्यविन्यास है।

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो आपको निम्न आउटपुट दिखाई देगा।

Laverne Hegmann (4e97c816-6bce-11e1-b095-6cf049ee52be)
   Points: 74
   HasGoldStatus: True
   MemberSince: 4/4/2009 12:00:00 AM (Utc)
   CreditRating: Neutral
   AverageRating: 0

   Orders:
      Order Id: 4ea14d96-6bce-11e1-b095-6cf049ee52be
      Order Id: 4ea14d96-6bce-11e1-b096-6cf049ee52be
      Order Id: 4ea14d96-6bce-11e1-b097-6cf049ee52be
      Order Id: 4ea14d96-6bce-11e1-b098-6cf049ee52be
		
Press <ENTER> to exit...

नुकसान में से एक यह है कि मान लें कि हम ऐसा कहना चाहते हैं FirstName.StartsWith(“A”) जैसा कि निम्नलिखित कार्यक्रम में दिखाया गया है।

var customers = session.QueryOver<Customer>() .Where(x => x.FirstName.StartsWith("A"));
 
foreach (var customer in customers.List()) { 
   Console.WriteLine(customer); 
} 

tx.Commit();

अब चलो फिर से एप्लिकेशन चलाते हैं और आप देखेंगे कि यह एक LINQ प्रदाता नहीं है क्योंकि यह नहीं जानता कि यह क्या है StartsWith विधि है, तो आप एक मिल जाएगा RunTime exception

अपवाद अपरिचित विधि कॉल को कहता है। यहाँ हम स्पष्ट काम कर रहे हैं, लेकिन यह जरूरी काम नहीं है।

चलिए कुछ और कोशिश करते हैं, जैसे कि फर्स्टनेम "A%" के बराबर है जैसा कि निम्नलिखित कोड में दिखाया गया है।

var customers = session.QueryOver<Customer>() .Where(x => x.FirstName == "A%"); 

foreach (var customer in customers.List()) { 
   Console.WriteLine(customer); 
}

चलो इसे एक बार फिर से चलाते हैं और आप देखेंगे कि हम नीचे दिखाए गए अनुसार कोई परिणाम प्राप्त नहीं करने जा रहे हैं।

Press <ENTER> to exit...

यह समझने के लिए कि हमें कोई परिणाम क्यों नहीं मिल रहा है, आइए NHibernate प्रोफाइलर पर एक नजर डालते हैं।

जैसा कि आप देख सकते हैं कि पहला नाम ए% के बराबर है जो नहीं है। SQL में%% का उपयोग ऑपरेटर की तरह किया जाता है। अब हमें निम्न कार्यक्रम में दिखाए गए अनुसार WHERE क्लॉज़ में एक प्रतिबंध बनाने की आवश्यकता है।

var customers = session.QueryOver<Customer>() 
   .Where(Restrictions.On<Customer>(c => c.FirstName).IsLike("A%")); 
	
foreach (var customer in customers.List()) { 
   Console.WriteLine(customer); 
}

आइए अपना एप्लिकेशन फिर से चलाएं और आप देखेंगे कि सभी ग्राहकों को पहले नाम A से शुरू किया गया है।

Alejandrin Will (4ea3aef6-6bce-11e1-b0b4-6cf049ee52be)
   Points: 24
   HasGoldStatus: False
   MemberSince: 10/1/2011 12:00:00 AM (Utc)
   CreditRating: VeryVeryGood
   AverageRating: 0

   Orders:
      Order Id: 4ea3aef6-6bce-11e1-b0b5-6cf049ee52be

Austyn Nolan (4ea871b6-6bce-11e1-b110-6cf049ee52be)
   Points: 67
   HasGoldStatus: True
   MemberSince: 12/29/2007 12:00:00 AM (Utc)
   CreditRating: Neutral
   AverageRating: 0

   Orders:
      Order Id: 4ea871b6-6bce-11e1-b111-6cf049ee52be

Antonia Murphy (4ea871b6-6bce-11e1-b121-6cf049ee52be)
   Points: 72
   HasGoldStatus: True
   MemberSince: 6/15/2009 12:00:00 AM (Utc)
   CreditRating: Terrible
   AverageRating: 0

   Orders:
      Order Id: 4ea871b6-6bce-11e1-b122-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b123-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b124-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b125-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b126-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b127-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b128-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b129-6cf049ee52be
      Order Id: 4ea871b6-6bce-11e1-b12a-6cf049ee52be

यह पहले की तरह ही काम करता है, इस नए प्रयोग को छोड़कर QueryOverवाक्य - विन्यास। कई डेवलपर्स पाते हैं कि LINQ सिंटैक्स अधिक अनुमानित है और अक्सर सही चीजें करता है।

यदि LINQ इसे संभाल नहीं सकता है, तो आप HQL या मानदंड को देखना शुरू कर देंगे कि क्या यह अधिक उपयुक्त होने वाला है।

यह आपको केवल एक अलग वाक्यविन्यास प्रदान करता है, इसलिए मानदंड, दोनों मानदंड और क्वेरीओवर आपको केवल एक और क्वेरी तंत्र प्रदान करते हैं जो आपको NHibernate का उपयोग करके डेटाबेस से डेटा खींचने की अनुमति देता है।