配置ABP框架使用對象映射
DTO和實(shí)體
實(shí)體
實(shí)體是領(lǐng)域驅(qū)動(dòng)設(shè)計(jì)(Domain Driven Design)中的概念,實(shí)體通常一一映射某些對象的固有屬性,最常使用的是關(guān)系型數(shù)據(jù)庫中的表。
在 ABP 中,實(shí)體位于領(lǐng)域?qū)又?,?shí)體類需要實(shí)現(xiàn) IEntity<TKey>
接口或繼承 Entity<TKey>
基類,示例如下:
public class Book : Entity<Guid> { public string Name { get; set; } public float Price { get; set; } }
DTO
數(shù)據(jù)傳輸對象(Data Transfer Object),作為數(shù)據(jù)傳輸過程中的數(shù)據(jù)模型,用于在應(yīng)用層和表示層之間傳輸數(shù)據(jù)。
在 ABP 中,DTO 位于應(yīng)用服務(wù)層,即本系列文章示例源碼中的 AbpBase.Application
項(xiàng)目。
通常表示層或其它類型的客戶端調(diào)用應(yīng)用服務(wù)時(shí),將 DTO 作為參數(shù)傳遞,它使用領(lǐng)域?qū)ο螅▽?shí)體)執(zhí)行某些特定的業(yè)務(wù)邏輯,并將 DTO (跟傳入的 DTO 不是同一個(gè))返回到表示層中,因此表示層與領(lǐng)域?qū)油耆綦x。
DTO 類 可能會跟 實(shí)體類的字段/屬性高度相似,為每個(gè)服務(wù)的每個(gè)方法創(chuàng)建 DTO 類可能會很枯燥且費(fèi)時(shí)間。
ABP 的 DTO 類示例如下:
public class ProductDto : EntityDto<Guid> { public string Name { get; set; } //... }
麻煩的映射
前面提到,領(lǐng)域?qū)雍蛻?yīng)用服務(wù)層是要隔離的,例如以下偽代碼:
class HomeController { AddService _service; [HttpPost] public int AddEquip(EquipDto dto) { return _service.Add(dto).Id; } } class AddService { DataContext _context; EquipDto Add(EquipDto dto) { Equip equip = new Equip() { Name = dto.Name; }; _context.Equip.Add(equip); _context.SaveChange(); dto.Id = equip.Id; return dto; } } class EquipDto { int Id; string Name; } ---------- class Equip { int Id; string Name; }
這樣每次都需要手動(dòng)為 DTO 類和 實(shí)體類手動(dòng)對字段賦值映射,當(dāng)一個(gè)實(shí)體有數(shù)十個(gè)字段時(shí),寫出的代碼會很冗長,而且容易忽略了某些字段,最終導(dǎo)致了 Bug。
大家都知道, AutoMapper 正好可以解決這個(gè)問題。
AutoMapper 集成
ABP 的 Volo.Abp.AutoMapper
模塊封裝或集成了 AutoMapper,所以我們正好使用模塊,為 ABP 應(yīng)用定義對象映射。
關(guān)于 AutoMapper 的使用,如何配置 Profile 等,筆者已經(jīng)單獨(dú)寫到 淺入 AutoMapper,請點(diǎn)擊鏈接另外學(xué)習(xí) AutoMapper 的使用。
我們可以在 AbpBase.Application
項(xiàng)目中,新建 一個(gè) AbpBaseApplicationAutoMapperProfile.cs
文件,這個(gè)文件用于實(shí)現(xiàn) Profile 以及定義映射。將服務(wù)領(lǐng)域的映射集中到這個(gè)文件中;或者新建一個(gè) Profiles
文件夾,在其中存放一些 Profile 類。
其內(nèi)容如下:
public class AbpBaseApplicationAutoMapperProfile:Profile { public AbpBaseApplicationAutoMapperProfile() { //base.CreateMap<MyEntity,MyDto>(); } }
定義完畢后,需要配置 AutoMapper 依賴注入,可在 AbpBaseApplicationModule
的 ConfigureServices
方法中,增加以下代碼:
Configure<AbpAutoMapperOptions>(options => { // 以模塊為單位注冊映射 options.AddMaps<AbpBaseApplicationModule>(); //// 以單個(gè) Profiel 為單位注冊映射 //options.AddProfile<AbpBaseApplicationAutoMapperProfile>(); });
在 Debug 階段,我們擔(dān)心項(xiàng)目改動(dòng)代碼時(shí),新增的字段忘記了加入到映射配置中,或者其它情況,在 AutoMapper 中,我們可以使用 configuration.AssertConfigurationIsValid();
來檢查映射;在 ABP 中則可使用 validate: true
參數(shù)來開啟檢查。
Configure<AbpAutoMapperOptions>(options => { // 以模塊為單位注冊映射 options.AddMaps<AbpBaseApplicationModule>(validate: true); //// 以單個(gè) Profiel 為單位注冊映射 //options.AddProfile<AbpBaseApplicationAutoMapperProfile>(validate: true); });
IObjectMapper/ObjectMapper
在 AbpBase.Application
項(xiàng)目中,添加 Nuget 包,搜索 Volo.Abp.ObjectMapping
并下載相應(yīng)的穩(wěn)定版本。
IObjectMapper 有兩個(gè),一個(gè)是 AutoMapper 的接口,一個(gè)是 Volo.Abp.ObjectMapping
的 泛型接口。
AutoMapper 的 IObjectMapper 不好用,所以別用;用 Volo.Abp.ObjectMapping
的 IObjectMapper <接口>
。
ObjectMapper 是 AutoMapper 中的,我們可以直接在控制器等位置,使用 ObjectMapper
注入,然后通過 ObjectMapper 實(shí)例映射對象。
ObjectMapper 只有 .Map()
這個(gè)方法用得順手。
private readonly ObjectMapper<T1,T2> _mapper; public TestController(ObjectMapper<T1,T2> mapper) { _mapper = mapper; // ... 使用示例 _ = mapper.Map<T1> (); }
也可以通過依賴注入使用 IObjectMapper
接口。
但是因?yàn)?ObjectMapper 是泛型類,每種類型的 DTO 都要注入一次的話,會很麻煩,因此這種方案也可以拋棄。
而 泛型的 IObjectMapper<TModule>
是一個(gè)抽象,我們使用 IObjectMapper<TModule>
做依賴注入的話,后續(xù)如果替換為別的對象映射框架,則不需要修改原有代碼即可完成替代。而且 IObjectMapper<TModule>
比較舒服。
使用示例:
private readonly IObjectMapper<AbpBaseApplicationModule> _mapper; public TestController(IObjectMapper<AbpBaseApplicationModule> mapper) { _mapper = mapper; // ... 使用示例 _ = mapper.Map<...>(); }
對象拓展
ABP框架提供了實(shí)體擴(kuò)展系統(tǒng)允許你添加額外屬性到已存在的對象 無需修改相關(guān)類。這句話是抄 ABP 官方文檔的。
要支持對象拓展映射,則需要開啟配置:
public class MyProfile : Profile { public MyProfile() { CreateMap<User, UserDto>() .MapExtraProperties(); } }
時(shí)間有限,筆者這里只把官方文檔的內(nèi)容講清楚,讀者看完后,需要繼續(xù)查閱官方文檔,完整了解對象拓展。
ObjectExtensionManager 是一個(gè)拓展對象映射類,可以顯式為類拓展一些額外的屬性,這個(gè)類型在 Volo.Abp.ObjectMapping
中定義。
ObjectExtensionManager 是一個(gè)類型,但是我們不能直接 new 它,或者使用依賴注入,只能通過 ObjectExtensionManager.Instance
這個(gè)屬性獲取新的類型。我們無需關(guān)心它是用了啥設(shè)計(jì)模式,還是因?yàn)榫彺嬷惖脑蜻@樣設(shè)計(jì)。
ObjectExtensionManager 有兩種屬性,其說明如下:
AddOrUpdate
:是定義對象額外屬性或更新對象額外屬性的主要方法;AddOrUpdateProperty
:快捷地定義單個(gè)拓展屬性的方法;
AddOrUpdateProperty
用于定義單個(gè)屬性,AddOrUpdate
是一個(gè)容器,可以包含多個(gè) AddOrUpdateProperty
。
AddOrUpdateProperty
示例代碼如下:
ObjectExtensionManager.Instance .AddOrUpdateProperty<TestA, string>("Name"); // 為 TestA 類添加了一個(gè) G 屬性
AddOrUpdate
的示例代碼如下:
ObjectExtensionManager.Instance .AddOrUpdate<TestA>(options => { options.AddOrUpdateProperty<string>("Name"); options.AddOrUpdateProperty<bool>("Nice"); } );
當(dāng)然,我們還可以同時(shí)為多個(gè)類型同時(shí)定義一個(gè)額外的屬性:
ObjectExtensionManager.Instance .AddOrUpdateProperty<string>( new[] { typeof(TestA), typeof(TestB), typeof(TestC) }, "Name" );
如果需要定義多個(gè)屬性,則可以使用 AddOrUpdate
:
ObjectExtensionManager.Instance .AddOrUpdate(options => { options.AddOrUpdateProperty<string>("Name"); }, new[]{ typeof(TestA), typeof(TestB) });
另外它還可以設(shè)置默認(rèn)值、增加驗(yàn)證規(guī)則等,這些筆者就不再贅述,讀者感興趣可以點(diǎn)擊鏈接進(jìn)入官方文檔查看。
https://docs.abp.io/zh-Hans/abp/latest/Object-Extensions#validation
到此這篇關(guān)于配置ABP框架使用對象映射的文章就介紹到這了。希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
asp .net實(shí)現(xiàn)給圖片添加圖片水印方法示例
圖片上加水印相信每位程序員都會遇到這個(gè)需求,下面這篇文章主要給大家介紹了asp .net實(shí)現(xiàn)給圖片添加圖片水印的方法,文中給出了完整的實(shí)例代碼,相信對大家具有一定的參考價(jià)值,需要的朋友們下面來一起看看吧。2017-03-03asp.net用url重寫URLReWriter實(shí)現(xiàn)任意二級域名
Asp.net 用url重寫(URLReWriter)實(shí)現(xiàn)任意二級域名2008-10-10.Net?Core應(yīng)用增強(qiáng)型跨平臺串口類庫CustomSerialPort()詳解
本文詳細(xì)講解了.Net?Core應(yīng)用增強(qiáng)型跨平臺串口類庫CustomSerialPort(),文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-01-01ASP.NET從字符串中查找字符出現(xiàn)次數(shù)的具體實(shí)現(xiàn)方法
今天在一場“特殊的討論”中引入了一個(gè)問題,如何在C#求出字符串中某字符的出現(xiàn)次數(shù),比如求“ADSFGEHERGASDF”中“A”出現(xiàn)的次數(shù)2013-11-11.net6簡單使用NPOI讀取Excel的項(xiàng)目實(shí)踐
本文主要介紹了.net6簡單使用NPOI讀取Excel的項(xiàng)目實(shí)踐,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06.NET中的MassTransit分布式應(yīng)用框架詳解
MassTransit是一款優(yōu)秀的分布式應(yīng)用框架,可作為分布式應(yīng)用的消息總線,也可以用作單體應(yīng)用的事件總線,這篇文章主要介紹了.NET中的MassTransit分布式應(yīng)用框架,需要的朋友可以參考下2022-10-10asp.net 參數(shù)不同共用一個(gè)頁面的實(shí)現(xiàn)方法
本文為大家介紹下asp.net參數(shù)不同如何共用一個(gè)頁面,感興趣的朋友不要錯(cuò)過2013-12-12