Dapper สนับสนุนคอลัมน์ที่กำหนดเองไปยังผู้ทำแผนที่ มันทำผ่านอินเตอร์เฟซITypeMap CustomPropertyTypeMapระดับให้บริการโดย Dapper ที่สามารถทำมากที่สุดของงานนี้ ตัวอย่างเช่น:
Dapper.SqlMapper.SetTypeMap(
typeof(TModel),
new CustomPropertyTypeMap(
typeof(TModel),
(type, columnName) =>
type.GetProperties().FirstOrDefault(prop =>
prop.GetCustomAttributes(false)
.OfType<ColumnAttribute>()
.Any(attr => attr.Name == columnName))));
และรูปแบบ:
public class TModel {
[Column(Name="my_property")]
public int MyProperty { get; set; }
}
เป็นสิ่งสำคัญที่จะต้องทราบว่าการใช้งาน CustomPropertyTypeMap นั้นต้องการคุณลักษณะที่มีอยู่และตรงกับหนึ่งในชื่อคอลัมน์มิฉะนั้นคุณสมบัติจะไม่ถูกแมป DefaultTypeMapระดับให้การทำงานที่ได้มาตรฐานและสามารถยกระดับการเปลี่ยนแปลงพฤติกรรมนี้:
public class FallbackTypeMapper : SqlMapper.ITypeMap
{
private readonly IEnumerable<SqlMapper.ITypeMap> _mappers;
public FallbackTypeMapper(IEnumerable<SqlMapper.ITypeMap> mappers)
{
_mappers = mappers;
}
public SqlMapper.IMemberMap GetMember(string columnName)
{
foreach (var mapper in _mappers)
{
try
{
var result = mapper.GetMember(columnName);
if (result != null)
{
return result;
}
}
catch (NotImplementedException nix)
{
// the CustomPropertyTypeMap only supports a no-args
// constructor and throws a not implemented exception.
// to work around that, catch and ignore.
}
}
return null;
}
// implement other interface methods similarly
// required sometime after version 1.13 of dapper
public ConstructorInfo FindExplicitConstructor()
{
return _mappers
.Select(mapper => mapper.FindExplicitConstructor())
.FirstOrDefault(result => result != null);
}
}
และเมื่อใช้งานแล้วมันจะกลายเป็นเรื่องง่ายในการสร้างตัวสร้างชนิดตัวกำหนดเองที่จะใช้คุณลักษณะโดยอัตโนมัติหากมีอยู่ แต่จะกลับไปใช้ลักษณะการทำงานแบบมาตรฐาน:
public class ColumnAttributeTypeMapper<T> : FallbackTypeMapper
{
public ColumnAttributeTypeMapper()
: base(new SqlMapper.ITypeMap[]
{
new CustomPropertyTypeMap(
typeof(T),
(type, columnName) =>
type.GetProperties().FirstOrDefault(prop =>
prop.GetCustomAttributes(false)
.OfType<ColumnAttribute>()
.Any(attr => attr.Name == columnName)
)
),
new DefaultTypeMap(typeof(T))
})
{
}
}
นั่นหมายความว่าขณะนี้เราสามารถสนับสนุนประเภทที่ต้องการแผนที่โดยใช้แอตทริบิวต์ได้อย่างง่ายดาย:
Dapper.SqlMapper.SetTypeMap(
typeof(MyModel),
new ColumnAttributeTypeMapper<MyModel>());
นี่คือสรุปสาระสำคัญการรหัสที่มาเต็ม