上一篇介紹了值對象的基本概念,得到了一些朋友的支持,另外也有一些朋友提出了不同意見。這其實是很自然的事情,設計本來就充滿了各種可能性,沒有絕對正確的做法,只有更好的實踐。但是設計與實踐的好與壞,對於不同的人,以及處於不同的環境都有不同的詮釋,這是一個仁者見仁,智者見智的問題。DDD非常抽象,以至於它的每一個概念,對於不同的人都有不同的看法,更何況基於DDD的.Net實踐,就更難分辨哪一個用法更標准、更正宗。
我對DDD的認識雖然還很膚淺,用得也很山寨,但這可能更加適合初步接觸DDD的朋友。還是那句老話,你不是搞學術研究的,你並不需要挖掘DDD的學術價值,而是要把它切實的用到你的項目上,並產生回報。你不應該問對或錯,而應該多看看哪些東西對你真正起作用,一方面需要多學習DDD理論知識,另一方面可以多參考其它人的用法,並琢磨出一套適合自己習慣的架構。特別是初學DDD的朋友,這一點更加重要,DDD水很深,盲目的采用某些你搞不懂的技術,只會增加負擔。你也不需要把DDD所有東西都用起來,使用DDD不是為了趕時髦,如果某些東西讓你感覺復雜,你先了解下就可以了,把搞懂的東西加入你的工具箱,然后項目上慢慢體驗,時間稍長,你就能產生突破並從中受益。但你如果人雲亦雲,把注意力放到純概念和一些名詞術語上,把別人的經驗生搬硬套到自己的項目,由於別人的思想你可能沒有真正搞懂,另外別人的項目需求、團隊水平、所用技術可能和你都不同,這樣可能導致你維護了一個龐大的架構,但卻沒有撈到一丁點好處。
我這個系列重點不在DDD,而是如何搭建自己的應用程序框架。介紹DDD分層架構只是保證本系列的完整性,所以我不會非常詳細的介紹。另外很多朋友迫切需要示例,我在此回復一下,本系列前期主要進行框架建設,包括一些公共操作類和層超類型,待底子打牢之后,我會向大家展示我的山寨DDD用法,以及如何通過應用程序框架快速開發項目。之所以不上來就搞一堆代碼,是希望你通過這個系列能真正受益,你不僅需要知道框架怎么用,更需要知道這玩意是怎么弄出來的,以及重要代碼的思考和演化過程。所以我寫得可能非常啰嗦,我希望.Net初學者也能看懂。我的時間比較有限,更新時間不會太快,不過只要有人願意繼續看,我會堅持寫完它。
下面回到正文上來,本篇將完成DDD值對象的層超類型開發,所有代碼都從網上搜集整理,如果大家有更好的請把你的代碼發上來供大家參考,另外最好詳細介紹你的代碼為何更好,以免大家憑空瞎猜。
首先,在Util.Domains類庫中創建一個名為ValueObjectBase的抽象類。
考慮值對象的相等性測試,怎樣才能認為兩個值對象是相等的?這可以通過比較兩個值對象的所有屬性值都相等來判斷,換句話說,兩個值對象有任何一個屬性值不同,都不相等。我們需要重寫Equals、GetHashCode 、==、!=這幾個方法或運算符。
在相等性比較中,我們可以通過反射來獲取所有屬性,並一一比較,以測試相等性。另外,GetHashCode將各屬性值的哈希碼使用簡單的異或操作計算出來。如果覺得性能不好,子類可以重寫相關實現。
另外,值對象有時候需要創建一個副本,可以增加一個克隆方法Clone,采用淺表復制進行創建,由於值對象不可變,所以不同的值對象共享相同的屬性值就不是什么問題。為了讓Clone更加好用,可以讓它創建出強類型的值對象,而不是一個object,這需要將值對象層超類型修改為泛型,將值對象作為泛型參數傳遞到基類。
另外,前面介紹的實體狀態輸出和驗證方法對值對象同樣適用,所以需要在實體和值對象層超類型之上再增加一個基類,命名為DomainBase。
由於值對象層超類型比較簡單,我就簡要介紹到這,下面是相關代碼,如有疑問請留言。
測試樣例Address類代碼如下,為了簡單,我只留下兩個屬性。
namespace Util.Domains.Tests.Samples { /// <summary>
/// 地址 /// </summary>
public class Address : ValueObjectBase<Address> { /// <summary>
/// 初始化地址 /// </summary>
/// <param name="city">城市</param>
/// <param name="street">街道</param>
public Address( string city, string street ) { City = city; Street = street; } /// <summary>
/// 城市 /// </summary>
public string City { get; private set; } /// <summary>
/// 街道 /// </summary>
public string Street { get; private set; } } }
值對象單元測試類ValueObjectBaseTest代碼如下。
using Microsoft.VisualStudio.TestTools.UnitTesting; using Util.Domains.Tests.Samples; namespace Util.Domains.Tests { /// <summary>
/// 值對象測試 /// </summary>
[TestClass] public class ValueObjectBaseTest { /// <summary>
/// 地址1 /// </summary>
private Address _address1; /// <summary>
/// 地址2 /// </summary>
private Address _address2; /// <summary>
/// 地址3 /// </summary>
private Address _address3; /// <summary>
/// 測試初始化 /// </summary>
[TestInitialize] public void TestInit() { _address1 = new Address("a","b"); _address2 = new Address( "a", "b" ); _address3 = new Address( "1", "" ); } /// <summary>
/// 測試對象相等性 /// </summary>
[TestMethod] public void TestEquals() { Assert.IsFalse( _address1.Equals( null ) ); Assert.IsFalse( _address1 == null ); Assert.IsFalse( null == _address1 ); Assert.IsFalse( _address1.Equals(new Test()) ); Assert.IsTrue( _address1.Equals( _address2 ), "_address1.Equals( _address2 )" ); Assert.IsTrue( _address1 == _address2, "_address1 == _address2" ); Assert.IsFalse( _address1 != _address2, "_address1 != _address2" ); Assert.IsFalse( _address1 == _address3, "_address1 == _address3" ); } /// <summary>
/// 測試哈希 /// </summary>
[TestMethod] public void TestGetHashCode() { Assert.IsTrue( _address1.GetHashCode() == _address2.GetHashCode(), "_address1.GetHashCode() == _address2.GetHashCode()" ); Assert.IsFalse( _address1.GetHashCode() == _address3.GetHashCode(), "_address1.GetHashCode() == _address3.GetHashCode()" ); } /// <summary>
/// 測試克隆 /// </summary>
[TestMethod] public void TestClone() { _address3 = _address1.Clone(); Assert.IsTrue( _address1 == _address3 ); } } }
DomainBase代碼如下。
using System.Collections.Generic; using System.Text; using Util.Validations; namespace Util.Domains { /// <summary>
/// 領域層頂級基類 /// </summary>
public abstract class DomainBase { #region 構造方法
/// <summary>
/// 初始化領域層頂級基類 /// </summary>
protected DomainBase() { _rules = new List<IValidationRule>(); _handler = new ValidationHandler(); } #endregion
#region 字段
/// <summary>
/// 描述 /// </summary>
private StringBuilder _description; /// <summary>
/// 驗證規則集合 /// </summary>
private readonly List<IValidationRule> _rules; /// <summary>
/// 驗證處理器 /// </summary>
private IValidationHandler _handler; #endregion
#region ToString(輸出領域對象的狀態)
/// <summary>
/// 輸出領域對象的狀態 /// </summary>
public override string ToString() { _description = new StringBuilder(); AddDescriptions(); return _description.ToString().TrimEnd().TrimEnd( ',' ); } /// <summary>
/// 添加描述 /// </summary>
protected virtual void AddDescriptions() { } /// <summary>
/// 添加描述 /// </summary>
protected void AddDescription( string description ) { if ( string.IsNullOrWhiteSpace( description ) ) return; _description.Append( description ); } /// <summary>
/// 添加描述 /// </summary>
protected void AddDescription<T>( string name, T value ) { if ( string.IsNullOrWhiteSpace( value.ToStr() ) ) return; _description.AppendFormat( "{0}:{1},", name, value ); } #endregion
#region SetValidationHandler(設置驗證處理器)
/// <summary>
/// 設置驗證處理器 /// </summary>
/// <param name="handler">驗證處理器</param>
public void SetValidationHandler( IValidationHandler handler ) { if ( handler == null ) return; _handler = handler; } #endregion
#region AddValidationRule(添加驗證規則)
/// <summary>
/// 添加驗證規則 /// </summary>
/// <param name="rule">驗證規則</param>
public void AddValidationRule( IValidationRule rule ) { if ( rule == null ) return; _rules.Add( rule ); } #endregion
#region Validate(驗證)
/// <summary>
/// 驗證 /// </summary>
public virtual void Validate() { var result = GetValidationResult(); HandleValidationResult( result ); } /// <summary>
/// 獲取驗證結果 /// </summary>
private ValidationResultCollection GetValidationResult() { var result = ValidationFactory.Create().Validate( this ); Validate( result ); foreach ( var rule in _rules ) result.Add( rule.Validate() ); return result; } /// <summary>
/// 驗證並添加到驗證結果集合 /// </summary>
/// <param name="results">驗證結果集合</param>
protected virtual void Validate( ValidationResultCollection results ) { } /// <summary>
/// 處理驗證結果 /// </summary>
private void HandleValidationResult( ValidationResultCollection results ) { if ( results.IsValid ) return; _handler.Handle( results ); } #endregion } }
ValueObjectBase代碼如下。
using System; using System.Linq; namespace Util.Domains { /// <summary>
/// 值對象 /// </summary>
/// <typeparam name="TValueObject">值對象類型</typeparam>
public abstract class ValueObjectBase<TValueObject> : DomainBase, IEquatable<TValueObject> where TValueObject : ValueObjectBase<TValueObject> { #region Equals(相等性比較)
/// <summary>
/// 相等性比較 /// </summary>
public bool Equals( TValueObject other ) { return this == other; } /// <summary>
/// 相等性比較 /// </summary>
public override bool Equals( object other ) { return Equals( other as TValueObject ); } #endregion
#region ==(相等性比較)
/// <summary>
/// 相等性比較 /// </summary>
public static bool operator ==( ValueObjectBase<TValueObject> valueObject1, ValueObjectBase<TValueObject> valueObject2 ) { if ( (object)valueObject1 == null && (object)valueObject2 == null ) return true; if ( (object)valueObject1 == null || (object)valueObject2 == null ) return false; if ( valueObject1.GetType() != valueObject2.GetType() ) return false; var properties = valueObject1.GetType().GetProperties(); return properties.All( property => property.GetValue( valueObject1 ) == property.GetValue( valueObject2 ) ); } #endregion
#region !=(不相等比較)
/// <summary>
/// 不相等比較 /// </summary>
public static bool operator !=( ValueObjectBase<TValueObject> valueObject1, ValueObjectBase<TValueObject> valueObject2 ) { return !( valueObject1 == valueObject2 ); } #endregion
#region GetHashCode(獲取哈希)
/// <summary>
/// 獲取哈希 /// </summary>
public override int GetHashCode() { var properties = GetType().GetProperties(); return properties.Select( property => property.GetValue( this ) ) .Where( value => value != null ) .Aggregate( 0, ( current, value ) => current ^ value.GetHashCode() ); } #endregion
#region Clone(克隆副本)
/// <summary>
/// 克隆副本 /// </summary>
public virtual TValueObject Clone() { return (TValueObject)MemberwiseClone(); } #endregion } }
.Net應用程序框架交流QQ群: 386092459,歡迎有興趣的朋友加入討論。
謝謝大家的持續關注,我的博客地址:http://www.cnblogs.com/xiadao521/
下載地址:http://files.cnblogs.com/xiadao521/Util.2014.11.27.1.rar