NHibernate - โหลด / รับ
ในบทนี้เราจะพูดถึงวิธีการทำงานของคุณสมบัติโหลดและรับและวิธีการใช้งาน นี่คือ API สองรายการที่คล้ายกันมากที่จัดทำโดยISession สำหรับการโหลดวัตถุด้วยคีย์หลัก
Get - มันจะส่งคืนวัตถุหรือโมฆะ
Load - มันจะส่งคืนวัตถุหรือจะโยนไฟล์ ObjectNotFoundException.
ทำไมเราถึงมี API สองตัวที่แตกต่างกัน
โหลด
เป็นเพราะ Load สามารถปรับฐานข้อมูลแบบไปกลับได้อย่างมีประสิทธิภาพมากขึ้น
Load ส่งคืนอ็อบเจ็กต์พร็อกซีและไม่จำเป็นต้องเข้าถึงฐานข้อมูลทันทีเมื่อคุณออก Load call นั้น
เมื่อคุณเข้าถึงพร็อกซีนั้นอ็อบเจ็กต์จะไม่อยู่ในฐานข้อมูลก็สามารถโยน ObjectNotFoundException ที่จุดนั้นได้
รับ
ตรงกันข้ามกับ Get เนื่องจากข้อ จำกัด ของ CLR หรือ Common Language Runtime และ NHibernate ต้องไปที่ฐานข้อมูลทันทีตรวจสอบว่ามีวัตถุอยู่ที่นั่นหรือไม่และส่งคืนค่าว่างหากไม่มีอยู่
ไม่มีตัวเลือกวัตถุในการหน่วงเวลาการดึงข้อมูลนั้นการเดินทางกลับไปยังฐานข้อมูลในภายหลังเนื่องจากไม่สามารถส่งคืนวัตถุพร็อกซีและเปลี่ยนวัตถุพร็อกซีนั้นเป็นค่าว่างเมื่อผู้ใช้เข้าถึงจริง
มาดูตัวอย่างง่ายๆซึ่งคุณจะเห็นว่าสิ่งเหล่านี้ถูกนำไปใช้จริงอย่างไรและความแตกต่างระหว่าง Get และ Load เราจะดำเนินการต่อด้วยคลาสโดเมนเดียวกันCustomers และ Orders และในทำนองเดียวกันไฟล์การแมปเดียวกันจากบทสุดท้าย
ในตัวอย่างนี้เราจะใช้ Get ดังที่แสดงในโปรแกรมต่อไปนี้ก่อน
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 id1 = Guid.Parse("4e97c816-6bce-11e1-b095-6cf049ee52be");
var id2 = Guid.Parse("AAAAAAAA-BBBB-CCCC-DDDD-EEEEEEEEEEEE");
var customer1 = session.Get<Customer>(id1);
Console.WriteLine("Customer1 data");
Console.WriteLine(customer1);
var customer2 = session.Get<Customer>(id2);
Console.WriteLine("Customer2 data");
Console.WriteLine(customer2);
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;
}
}
}
อย่างที่คุณเห็นว่าเรามีสอง GuidID แรกคือ ID ที่ดีเป็น ID ของลูกค้าที่เรารู้ว่าอยู่ในฐานข้อมูล แม้ว่า ID ที่สองจะไม่มีอยู่ในฐานข้อมูล ID ทั้งสองนี้ถูกส่งผ่านเป็นพารามิเตอร์ไปยังGet() วิธีการจากนั้นผลลัพธ์จะถูกพิมพ์บนคอนโซล
เมื่อโค้ดด้านบนถูกคอมไพล์และดำเนินการคุณจะเห็นผลลัพธ์ต่อไปนี้
Customer1 data
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
Customer2 data
Press <ENTER> to exit...
ดังที่คุณเห็นว่ามีการพิมพ์ข้อมูล Customer1 แต่ข้อมูล Customer2 ว่างเปล่านั่นเป็นเพราะไม่มีเรกคอร์ด Customer2 ในฐานข้อมูล
เมื่อคุณเรียกใช้แอปพลิเคชันของคุณอีกครั้งเราสามารถแทรกจุดพักก่อนคำสั่งคอมมิตจากนั้นให้ดูลูกค้าทั้งสองในหน้าต่าง Watch
ดังที่คุณเห็นว่ามีข้อมูล Customer1 ในขณะที่ Customer2 เป็นค่าว่างและประเภทคือ NHibernateDemo.Customer สำหรับทั้ง.
ตอนนี้เรามาใช้วิธีการโหลดแทน Get ในตัวอย่างเดียวกับที่แสดงในโค้ดต่อไปนี้
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 id1 = Guid.Parse("4e97c816-6bce-11e1-b095-6cf049ee52be");
var id2 = Guid.Parse("AAAAAAAA-BBBB-CCCC-DDDD-EEEEEEEEEEEE");
var customer1 = session.Load<Customer>(id1);
Console.WriteLine("Customer1 data");
Console.WriteLine(customer1);
var customer2 = session.Load<Customer>(id2);
Console.WriteLine("Customer2 data");
Console.WriteLine(customer2);
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;
}
}
}
ตอนนี้ให้เรียกใช้ตัวอย่างนี้และคุณจะเห็นว่ามีข้อยกเว้นต่อไปนี้ปรากฏขึ้นตามที่เห็นในภาพหน้าจอ
ตอนนี้ถ้าคุณดูที่หน้าต่าง Watch คุณจะเห็นประเภทคือพร็อกซีของลูกค้าสำหรับวัตถุทั้งสอง และคุณจะเห็นข้อมูลเดียวกันสำหรับ Customer1 บนหน้าต่างคอนโซล
Customer1 data
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
Customer2 data