假设你的pg数据库有这样一个table结构
test{
integer[] arr
}
如果用dapper查询ids,那么会返回一个int[]类型
如果你同时用protobuf生成了消息文件,
{
repeated int32 arr
}
他在程序中实际为一个集合类型(IList类型)
我阅读了Protobuf相关的代码和Dapper相关代码,
尝试重载操作符转换[] 为list
但是发现reapted 字段属性只有get没有set.
曾经一度想改protobuf或者dapper的源代码进行适应,但是权衡后发现并不合适,
突然想起来protobuf生成的文件为partial类.(部分类)
顿时醍醐灌顶,思如泉涌.
由于proto文件生成的类会自动首字母大写.
而postgresql对列名和表名区分大小写,所以
proto文件生成的列名为Arr
而我们数据库里的列名为arr
写一个扩展类
public sealed partial class test
{
public RepeatedField<int> arr {
get { return arr_; }
set { arr_.AddRange(value); }
}
}
再写一个Dapper的自定义列处理函数,如果要处理多种数组类型,那么把int改为T即可.
public class myCustomTypeCast : SqlMapper.TypeHandler<RepeatedField<int>>
{
public new void SetValue(IDbDataParameter parameter, int[] value)
{
parameter.Value = value;
}
public override void SetValue(IDbDataParameter parameter, RepeatedField<int> value)
{
parameter.Value = value;
}
public override RepeatedField<int> Parse(object value)
{
var tmp = value as IEnumerable<int>;
var result = new RepeatedField<int>();
result.AddRange(tmp);
Console.WriteLine("###1");
return result;
}
}
//只需要设置一次该类型处理函数即可
SqlMapper.AddTypeHandler(typeof(RepeatedField<int>), new myCustomTypeCast());
以上属于一种解决思路,
缺陷是如果你原来有一个Arr变量,现在会多一个arr变量.但好消息是,Arr和arr都是指向的一个对象,
他们的内容是一模一样的,因为他们只是作为一个属性访问器存在,Arr没有set方法,arr的set方法也并不实例化任何对象,他做的就是在目标对象上调用添加函数,将数据存放到数据应该去的地方!
enjoy it...