標籤:blog http io 使用 ar strong for 檔案 資料
本節內容
- 引入
- 方案1:直接添加
- 方案2:巧用組件
- 執行個體分析
- 結語
引入
通過前面7篇的學習,有點乏味了~~~這篇來學習一個技巧,大家一起想想如果我要在Customer類中實現一個Fullname屬性(就是Firstname和Lastname的組合)該怎麼做呢?
方案1:直接添加
“我知道!修改Customer類,添加一個Fullname屬性!即Customer.Fullname!”
“恩,完全正確......”
“這就意味著在Customer類中把Firstname和Lastname兩個屬性重新修改組合為Fullname屬性。這樣的話,如果有其它的類(像Vendor、Shiper)使用了Firstname和Lastname兩個屬性,這就需要修改很多商務邏輯。那你的麻煩可就大了,還有什麼方法嗎?”
“.........”
方案2:巧用組件
NHibernate中,提供了組件(Component)和動態組件來協助我們完成這件事情。其實組件在NHibernate中為了不同目的被重複使用。這裡我們使用它來依賴對象。
對應檔中,<component>元素把子物件的一些屬性對應為父類對應的表的一些欄位。然後,組件可以定義它們自己的屬性、組件或者集合。
下面用兩幅圖顯示組件和動態組件兩個節點映射屬性:
看看這些映射屬性:
- access(預設property):NHibernate用來訪問屬性的策略
- class(預設通過反射得到的屬性類型):組件(子)類的名字
- insert:被映射的欄位是否出現在SQL的INSERT語句中
- name:屬性名稱propertyName
- update:被映射的欄位是否出現在SQL的UPDATE語句中
- <property>子項目:為組件(子)類的一些屬性與表欄位之間建立映射
- <parent>子項目:在組件類內部就可以有一個指向其容器的實體的反向引用
<dynamic-component>元素允許一個IDictionary作為組件映射,其中屬性名稱對應字典中的鍵。這又是使用組件的另一種用法。
知道上面的知識,我們該想想上面的問題該如何利用組件來實現了吧。
執行個體分析
我們用一幅圖來展示我們這節所說的一切:
開始動手吧!
1.建立Name類
namespace DomainModel.Entities{ public class Name { public string Firstname { get; set; } public string Lastname { get; set; } public string Fullname { get { return Firstname + " " + Lastname; } } }}
簡單的說,這個類用於組合Fullname屬性。
2.修改Customer類
namespace DomainModel.Entities{ public class Customer { public virtual int CustomerId { get; set; } public virtual int Version { get; set; } public virtual Name Name { get; set; } }}
修改Customer類,去除原來的Firstname和Lastname屬性,添加Name屬性。這時Name作為Customer的一個組成部分。需要注意的是:和原來Firstname和Lastname屬性一樣,需要對Name的持久化屬性定義getter和setter方法,但不需要實現任何的介面或宣告身份識別符欄位。
3.修改Customer映射
<?xml version="1.0" encoding="utf-8" ?><hibernate-mapping xmlns="urn:nhibernate-mapping-2.2" assembly="DomainModel" namespace="DomainModel"> <class name ="DomainModel.Entities.Customer,DomainModel" table="Customer"> <id name="CustomerId" column="CustomerId" type="Int32" unsaved-value="0"> <generator class ="native"></generator> </id> <version name="Version" column="Version" type="integer" unsaved-value="0"/> <component name="Name" class="DomainModel.Entities.Name,DomainModel"> <property name="Firstname" column ="Firstname" type="string" length="50" not-null="false" unique-key="UC_CustomerName"/> <property name ="Lastname" column="Lastname" type="string" length="50" not-null="false" unique-key="UC_CustomerName"/> </component> </class></hibernate-mapping>
首先定義Component的一些屬性,指定屬性名稱和組件映射的類名。再使用<property>子項目,為Name類的Firstname、Lastname屬性與表欄位之間建立映射。是不是很簡單~~
這時Customer表中還是CustomerId、Version、Firstname、Lastname欄位。完全不需要修改資料庫表結構哦。
這裡需要注意兩點:
- 就像所有的實值型別一樣,組件不支援共用引用。組件的值為空白從語意學上來講是專有的。每當重新載入一個包含組件的對象,如果組件的所有欄位為空白,那麼NHibernate將假定整個組件為空白。對於絕大多數目的,這樣假定是沒有問題的。
- 組件的屬性可以是NHibernate類型(包括集合、多對一關聯以及其它組件)。嵌套組件不應該作為特殊的應用被考慮。NHibernate趨向於支援設計細粒度的物件模型。
4.編寫方法
這時,我們需要修改或者重新編寫新的方法來實現我們想要的邏輯。
public IList<Customer> ReturnFullName(string firstname, string lastname){ return _session .CreateQuery("select from Customer c where c.Name.Firstname=:fn and c.Name.Lastname=:ln") .SetString("fn", firstname) .SetString("ln", lastname) .List<Customer>();}
現在,我們訪問Customer的Firstname、Lastname屬性,只需要在原來的基礎上通過Name訪問,例如上面修改的情況,看看上面圖片上怎麼訪問的吧,一目瞭然。
如果我們要添加一個Customer怎麼辦呢?程式碼片段如下所示:
var customer = new Customer() { Name = new Name() { Firstname = "YJing", Lastname = "Lee" } };
5.測試方法
有了上面的方法,我們編寫一個測試案例測試一下這個方法吧:看看結果測試成功,OK。
[Test]public void ReturnFullNameTest(){ IList<Customer> customers = _relation.ReturnFullName("YJing","Lee"); foreach (Customer c in customers) { Assert.AreEqual("YJing Lee", c.Name.Fullname); }}
結語
這一篇像大家介紹一個使用組件技巧,通過組件可以改善我們的物件模型,而資料庫結構不需要變化。通過這一篇的技巧,利用組件來映射來依賴對象,可以非常連貫的引入NHibernate中的多表映射關係、集合等內容,這些才是NHibernate中的亮點,就連LINQ都比不過它。從下篇開始就來學習NHibernate中的閃光點。
[轉]NHibernate之旅(8):巧用組件之依賴對象