自定义类型转换
有时,需要完全控制一个类型到另一个类型的转换。一个类型一点都不像另一个类型,而且转换函数已经存在了,在这种情况下,你想要从一个“宽松”的类型转换成一个更强壮的类型,例如一个string的源类型到一个int32的目标类型。
这里有两个类Source和Destination,要把前者映射到后者,代码如下:
public class Source { public string Value1 { get; set; } public string Value2 { get; set; } public string Value3 { get; set; } }
public class Destination { public int Value1 { get; set; } public DateTime Value2 { get; set; } public Type Value3 { get; set; } }
截至发稿前,官方文档这样描述的“因为AutoMapper不清楚从string到int,Datetime或Type的映射,如果要尝试映射的话,AutoMapper就会抛出异常(在映射时和配置检查时)”。为了创建 这些类型的映射,我们必须提供自定义的类型转换器,我们有三种方法这样做:
void ConvertUsing(Func<TSource, TDestination> mappingFunction); void ConvertUsing(ITypeConverter<TSource, TDestination> converter); void ConvertUsing<TTypeConverter>() where TTypeConverter : ITypeConverter<TSource, TDestination>;
但是,我先不实现自定义的类型转换器试一下:
namespace ThirdAutoMapper { class Program { static void Main(string[] args) { Mapper.CreateMap<Source, Destination>(); var source = new Source { Value1 = "5", Value2 = "05/11/2015", Value3 = "ThirdAutoMapper.Source" }; var destination = Mapper.Map<Destination>(source); Console.WriteLine("destination.Value1={0}", destination.Value1); Console.WriteLine("destination.Value2={0}", destination.Value2); Console.Read(); } } }
出现如下错误,但从错误看来,只有从string到Type类型映射时,出现了错误,其他两种类型呢?
现在我们将源类型和目标类型的第三个属性Value3注释掉,同时注释掉参与映射前的源类型对象的Value3属性,再来试一下。
果然,发现映射成功。也就说说最新版的AutoMapper默认可以将string类型映射为int和DateTime类型了。但是string还是没办法映射为Type类型。
对于以上AutoMapper的API给出的三种转换方法,第一个选择只是输入一个源返回一个目标的函数,这对于简单场合是有效的,但是对于更大的案例会变得难以处理。在更复杂的情况下,我们可以一个自定义的类型转换器,通过实现ITypeConverter<TSource, TDestination>接口创建:
第一种方法:
public class CustomTypeConverter : ITypeConverter<Source, Destination> { public Destination Convert(ResolutionContext context) { Source src = context.SourceValue as Source; var dest = new Destination { Value1 = System.Convert.ToInt32(src.Value1), Value2 = System.Convert.ToDateTime(src.Value2), Value3 = context.SourceType }; return dest; } }
然后给AutoMapper提供一个自定义类型的转换器或者AutoMapper在运行时能实例化的简化类型。对于上面的源/目标类型的映射配置就得以实现了:
修改Main方法中的代码为:
static void Main(string[] args) { Mapper.CreateMap<Source, Destination>().ConvertUsing<CustomTypeConverter>(); var source = new Source { Value1 = "5", Value2 = "05/11/2015", Value3 = typeof(Source).ToString()
}; var destination = Mapper.Map<Destination>(source); Console.WriteLine("destination.Value1={0}", destination.Value1); Console.WriteLine("destination.Value2={0}", destination.Value2); Console.WriteLine(destination.Value3.ToString()==source.Value3); Console.WriteLine(destination.Value3); Console.Read(); }
测试结果如下,映射成功!
第二种方法:
每个类型分别实现ITypeConverter接口:
public class DateTimeTypeConverter:ITypeConverter<string,DateTime> { public DateTime Convert(ResolutionContext context) { return System.Convert.ToDateTime(context.SourceValue); } } public class TypeTypeConverter : ITypeConverter<string, Type> { public Type Convert(ResolutionContext context) { return context.SourceValue == typeof(Source).ToString() ? typeof(Source) : typeof(Destination); } }
static void Main(string[] args) { //我看网上很多这种写法,包括官方文档,但是我这样写编译错误,可能是现在的AutoMapper不支持了吧 // Mapper.CreateMap<string, int>().ConvertUsing( Convert.ToInt32); Mapper.CreateMap<string, int>().ConvertUsing((context, intNum) => Convert.ToInt32(context.SourceValue)); Mapper.CreateMap<string, DateTime>().ConvertUsing(new DateTimeTypeConverter().Convert); Mapper.CreateMap<string, Type>().ConvertUsing<TypeTypeConverter>(); Mapper.CreateMap<Source, Destination>(); //Mapper.CreateMap<Source, Destination>().ConvertUsing<CustomTypeConverter>(); var source = new Source { Value1 = "5", Value2 = "05/11/2015", Value3 = typeof(Source).ToString() }; var destination = Mapper.Map<Destination>(source); Console.WriteLine("destination.Value1={0}", destination.Value1); Console.WriteLine("destination.Value2={0}", destination.Value2); Console.WriteLine(destination.Value3.ToString()==source.Value3); Console.WriteLine(destination.Value3); Console.Read(); }
测试结果,同样完美通过。