postgresql用dapper操作并且联合protobuf的array数组问题

假设你的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 arr {
        get { return arr_; }
        set { arr_.AddRange(value); }
    }
}

再写一个Dapper的自定义列处理函数,如果要处理多种数组类型,那么把int改为T即可.

public class myCustomTypeCast : SqlMapper.TypeHandler>
{
    public new void SetValue(IDbDataParameter parameter, int[] value)
    {
        parameter.Value = value;
    }

    public override void SetValue(IDbDataParameter parameter, RepeatedField value)
    {
        parameter.Value = value;
    }

    public override RepeatedField Parse(object value)
    {
        var tmp = value as IEnumerable;
        var result = new RepeatedField();
        result.AddRange(tmp);
        Console.WriteLine("###1");
        return result;

    }
}

//只需要设置一次该类型处理函数即可
SqlMapper.AddTypeHandler(typeof(RepeatedField), new myCustomTypeCast());

以上属于一种解决思路,

缺陷是如果你原来有一个Arr变量,现在会多一个arr变量.但好消息是,Arr和arr都是指向的一个对象,

他们的内容是一模一样的,因为他们只是作为一个属性访问器存在,Arr没有set方法,arr的set方法也并不实例化任何对象,他做的就是在目标对象上调用添加函数,将数据存放到数据应该去的地方!

enjoy it...

你可能感兴趣的:(c#,postgresql,protobuf,dapper)