AutoMapper(七)

tkbSimplest發表於2015-11-07

返回總目錄


Null值替換

如果源型別的成員鏈上的屬性值為Null,Null值替換允許提供一個可替換的值。下面有兩個類Person和PersonInfo類,都有一個屬性Title(頭銜),從Person對映到PersonInfo,如果Person的屬性沒有賦值,那麼PersonInfo的對應屬性值就用“屌絲”來替換。

namespace SeventhAutoMapper
{
    class Person
    {
        public string Title { get; set; } 
    }

    class PersonInfo
    {
        public string Title { get; set; }
    }
    class Program
    {
        static void Main(string[] args)
        {
            //對映
            Mapper.CreateMap<Person, PersonInfo>()
                .ForMember(dest => dest.Title, opt => opt.NullSubstitute("屌絲"));//源屬性如果為null,置為“屌絲”
            //執行對映
            var personInfo = Mapper.Map<PersonInfo>(new Person());//源屬性未賦值,故為null
          var personInfo2 = Mapper.Map<PersonInfo>(new Person(){Title = "高富帥"});//源屬性有值
            //輸出結果
            Console.WriteLine("personInfo.Title=" + personInfo.Title);
          Console.WriteLine("personInfo2.Title=" + personInfo2.Title);
          Console.Read();
        }
    }
}

測試結果如下:

image

開放泛型

AutoMapper支援開放泛型對映。下面建立兩個泛型類:

class Soure<T>
{
    public T Value { get; set; }
}

class Destination<T>
{
    public T Value { get; set; }
}

我們不需要建立封閉的泛型型別(也就是不將具體的型別寫死在尖括號內),AutoMapper在執行時會將開放泛型的任何配置應用到關閉的對映上去。

//建立開放泛型的對映
Mapper.CreateMap(typeof(Source<>),typeof(Destination<>));

var src1 = new Source<int> {Value = 22};
var dest1= Mapper.Map<Destination<int>>(src1);
Console.WriteLine(dest1.Value);

var src2  = new Source<string> {Value = "Hello,AutoMapper!"};
var dest2 = Mapper.Map<Destination<string>>(src2);
Console.WriteLine(dest2.Value);

//......依次類推
Console.Read();

測試結果如下:

image

因為C#只允許關閉的泛型形參,所以必須使用不帶泛型引數的CreateMap方法來建立自己的開放泛型引數對映,同時可以使用所有可以利用的對映配置。AutoMapper在配置驗證期間會跳過開放泛型型別對映。

也可以建立一個開放泛型轉換器:

Mapper.CreateMap(typeof(Source<>), typeof(Destination<>)).ConvertUsing(typeof(Converter<>));

投影

通過扁平化物件模型,將一個源型別轉換成一個目標型別。不需要額外的配置,AutoMapper只要求一個扁平的目標型別能匹配源型別的命名結構。當把一個源值投影到一個不精準匹配源結構的目標值時,必須指明成員對映定義。

舉個栗子,我們想把一個源結構CalendarEvent轉成一個在Web頁面上方便使用者輸入的目標結構CalendarEventForm:

public class CalendarEvent
{
    public DateTime Date { get; set; }
    public string Title { get; set; }
}

public class CalendarEventForm
{
    public DateTime EventDate { get; set; }
    public int EventHour { get; set; }
    public int EventMinute { get; set; }
    public string Title { get; set; }
}

由於目標屬性不是很匹配源屬性(CalendarEvent.Date需要成為CalendarEventForm.EventDate),我們需要在型別對映配置中指明成員的對映規則:

var calender = new CalendarEvent()
{
    Date = DateTime.Now,
    Title = "歷史性時刻"
};
//建立對映
Mapper.CreateMap<CalendarEvent, CalendarEventForm>()
    .ForMember(dest => dest.EventDate, opt => opt.MapFrom(src => src.Date.Date))
    .ForMember(dest => dest.EventHour, opt => opt.MapFrom(src => src.Date.Hour))
    .ForMember(dest => dest.EventMinute, opt => opt.MapFrom(src => src.Date.Minute));
//執行對映
var calenderForm = Mapper.Map<CalendarEventForm>(calender);
//輸出對映前的物件
Console.WriteLine("calender.Date={0},Title={1}",calender.Date,calender.Title);
//輸出對映後的物件
foreach (PropertyInfo info in calenderForm.GetType().GetProperties())
{
    Console.WriteLine(info.Name+"="+info.GetValue(calenderForm));
}

測試結果如下:

image

 

好的,關於AutoMapper的系列教程就先告一段落了,當時開這系列教程的初衷是為了順利地進行專案地開發,現在會用了,關於AutoMapper的話題就暫時先放放,不過以後肯定還會有關於AutoMapper的部落格的。

相關文章