NHibernate - व्युत्क्रम संबंध

इस अध्याय में, हम एक अन्य विशेषता को शामिल करेंगे जो कि व्युत्क्रम संबंध है। यह एक मनोरंजक विकल्प है जिसे आप संग्रह पर देखेंगे जो कि सत्य के विपरीत हैं और यह बहुत सारे डेवलपर्स को भ्रमित करता है। तो चलिए इस विकल्प के बारे में बात करते हैं। इसे समझने के लिए, आपको वास्तव में संबंधपरक मॉडल के बारे में सोचना होगा। मान लीजिए कि आपके पास एक एकल विदेशी कुंजी का उपयोग करके एक द्विदिश संघ हैं।

  • संबंधपरक दृष्टिकोण से, आपको एक विदेशी कुंजी मिली है, और यह ग्राहक को ऑर्डर करने और ऑर्डर करने के लिए दोनों ग्राहक का प्रतिनिधित्व करता है।

  • OO मॉडल से, आपके पास इन संदर्भों का उपयोग करने वाले यूनिडायरेक्शनल एसोसिएशन हैं।

  • ऐसा कुछ भी नहीं है जो कहता है कि दो यूनिडायरेक्शनल एसोसिएशन डेटाबेस में एक ही द्विदिश एसोसिएशन का प्रतिनिधित्व करते हैं।

  • यहाँ समस्या यह है कि NHibernate को यह जानने के लिए पर्याप्त जानकारी नहीं है customer.orders तथा order.customer डेटाबेस में एक ही संबंध का प्रतिनिधित्व करते हैं।

  • हमें प्रदान करने की आवश्यकता है inverse equals true एक संकेत के रूप में, यह इसलिए है क्योंकि यूनिडायरेक्शनल एसोसिएशन एक ही डेटा का उपयोग कर रहे हैं।

  • अगर हम इन रिश्तों को बचाने की कोशिश करते हैं, जिसमें उनके 2 संदर्भ हैं, तो NHibernate उस संदर्भ को दो बार अपडेट करने की कोशिश करेगा।

  • यह वास्तव में डेटाबेस के लिए एक अतिरिक्त राउंडट्रिप करेगा, और इसमें उस विदेशी कुंजी के 2 अपडेट भी होंगे।

  • उलटा सच NHibernate बताता है कि रिश्ते के किस पक्ष को अनदेखा करना है।

  • जब आप इसे संग्रह पक्ष पर लागू करते हैं और NHibernate हमेशा दूसरी तरफ से चाइल्ड ऑब्जेक्ट साइड से विदेशी कुंजी को अपडेट करेगा।

  • तब हमारे पास उस विदेशी कुंजी का केवल एक अपडेट होता है और हमारे पास उस डेटा के अतिरिक्त अपडेट नहीं होते हैं।

  • यह हमें विदेशी कुंजी को इन डुप्लिकेट अपडेट को रोकने की अनुमति देता है और इससे हमें विदेशी कुंजी उल्लंघन को रोकने में भी मदद मिलती है।

आइए नजर डालते हैं customer.cs फ़ाइल जिसमें आप देखेंगे AddOrderयहाँ विधि और विचार यह है कि अब हमारे पास ऑर्डर बैक से ग्राहक तक यह बैक पॉइंटर है और इसे सेट करने की आवश्यकता है। इसलिए जब किसी ग्राहक के लिए एक आदेश जोड़ा जाता है, तो उस ग्राहक का बैक पॉइंटर सेट किया जाता है, अन्यथा, यह अशक्त होगा, इसलिए हमें इसे ऑब्जेक्ट ग्राफ में एक साथ ठीक से कनेक्ट रखने के लिए इसकी आवश्यकता है।

using System; 
using System.Text; 
using Iesi.Collections.Generic;

namespace NHibernateDemo {
 
   public class Customer { 
      
      public Customer() {
         MemberSince = DateTime.UtcNow; Orders = new HashedSet<Order>();
      } 
      
      public virtual Guid Id { get; set; } 
      public virtual string FirstName { get; set; } 
      public virtual string LastName { get; set; } 
      public virtual double AverageRating { get; set; } 
      public virtual int Points { get; set; } 
      public virtual bool HasGoldStatus { get; set; } 
		
      public virtual DateTime MemberSince { get; set; } 
      public virtual CustomerCreditRating CreditRating { get; set; } 
      public virtual Location Address { get; set; }
      public virtual ISet<Order> Orders { get; set; }
      public virtual void AddOrder(Order order) { Orders.Add(order); order.Customer = this; }
      
      public override string ToString() { 
         var result = new StringBuilder(); 
			
         result.AppendFormat("{1} {2} ({0})\r\n\tPoints: {3}\r\n\tHasGoldStatus:
            {4}\r\n\tMemberSince: {5} ({7})\r\n\tCreditRating: {6}\r\n\tAverageRating:
            {8}\r\n", Id, FirstName, LastName, Points, HasGoldStatus, MemberSince,
            CreditRating, MemberSince.Kind, AverageRating);
         result.AppendLine("\tOrders:"); 
         
         foreach(var order in Orders) { 
            result.AppendLine("\t\t" + order); 
         } 
			
         return result.ToString(); 
      } 
   }
   
   public class Location { 
      public virtual string Street { get; set; } 
      public virtual string City { get; set; } 
      public virtual string Province { get; set; } 
      public virtual string Country { get; set; }
   } 
   
   public enum CustomerCreditRating { 
      Excellent, 
      VeryVeryGood, 
      VeryGood, 
      Good, 
      Neutral, 
      Poor, 
      Terrible 
   } 
}

यहाँ है 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 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(); 
         }
			
         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 खोलें और देखें कि वास्तव में इसे कैसे बचाया।

आप देखेंगे कि हमारे पास बयानों के 3 समूह हैं। पहले एक ग्राहक को सम्मिलित करेगा, और उस ग्राहक की आईडी मार्गदर्शिका है, जिसे हाइलाइट किया गया है। दूसरा बयान आदेश तालिका में डाला गया है।

आपको लगता है कि वही ग्राहक आईडी गाइड वहां सेट है, इसलिए उस विदेशी कुंजी को सेट करें। अंतिम विवरण अद्यतन है, जो एक बार फिर उसी ग्राहक आईडी के लिए विदेशी कुंजी को अपडेट करेगा।

अब समस्या यह है कि ग्राहक के पास आदेश हैं, और आदेशों के पास ग्राहक है, कोई रास्ता नहीं है कि हमने NHibernate को नहीं बताया है कि यह वास्तव में एक ही संबंध है। जिस तरह से हम ऐसा करते हैं, उलटा सच के साथ होता है।

तो चलिए हमारे पास 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" 
         inverse = "true"> 
         <key column = "CustomerId"/> 
         <one-to-many class = "Order"/> 
      </set> 
   
   </class> 
</hibernate-mapping>

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

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

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

  • यदि आप एक से कई संबंधों को देखते हैं Order.hbm.xml फ़ाइल और आप उलटा खोजते हैं, इसमें वास्तव में व्युत्क्रम विशेषता नहीं होती है।

  • यह हमेशा चाइल्ड आइटम से सेट किया जाता है, लेकिन यदि आपके पास कई-से-कई संग्रह हैं, तो आप इसे दोनों ओर से सेट कर सकते हैं।