postgresql 函数 参数为复合类型

postgresql没有存储过程,但是函数功能很强大、

在近期开发的电商管理平台中,对于产品的类目管理,设计时有个属性字段,设为字符数组,但是EF不支持数组的操作,所以在添加和修改类目时,需要对属性的存储和维护进行特殊处理,导致代码量增大。

public static async Task AddCategory(MerchandiseCategory category)
{
using (MasonDB db = new MasonDB())
{
var p_attributes = new NpgsqlParameter("attributes", NpgsqlDbType.Array | NpgsqlDbType.Text);
p_attributes.Value = category.Attributes;
var p_parentid = new NpgsqlParameter { ParameterName = "parentid", Value = category.ParentID };
var p_categoryname = new NpgsqlParameter { ParameterName = "categoryname", Value = category.CategoryName };
var p_isvalid = new NpgsqlParameter { ParameterName = "isvalid", Value = category.IsValid };
var p_operatorid = new NpgsqlParameter { ParameterName = "operatorid", Value = category.OperatorID };
var p_operatortime = new NpgsqlParameter { ParameterName = "operatortime", Value = category.OperateTime };
var p_sortweight = new NpgsqlParameter { ParameterName = "sortweight", Value = DateTime.Now.Subtract(DateTime.Parse("1970-1-1")).TotalMilliseconds };
await db.Database.ExecuteSqlCommandAsync("INSERT INTO \"MerchandiseCategories\" (\"ParentID\", \"CategoryName\", \"IsValid\", \"OperatorID\", \"OperateTime\", \"SortWeight\", \"Attributes\") VALUES(@parentid, @categoryname, @isvalid, @operatorid, @operatortime,@sortweight,@attributes); ", p_parentid, p_categoryname, p_isvalid, p_operatorid, p_operatortime, p_sortweight, p_attributes);

后期如果需要对数据库表加入新的字段或者是删除不需要的字段,SQL作为弱类型语言就会暴露出很多问题、编译无法通过,我们修改代码后重启服务器,这带来的后果有时是不可估量的。所以在开发中采用EF和LINQ。

为了减少代码量,以及便于维护,因此把插入和修改都用数据库函数实现、

postgresql函数的参数可以是表类型、即复合类型、要注意的是表类型要加“”,这个和oracle是一样的、

插入时我想得到插入的一行的数据的ID,代码实现如下:

CREATE OR REPLACE FUNCTION func_add_merchandisecategory(category "MerchandiseCategories")
RETURNS integer AS
$BODY$
INSERT INTO "MerchandiseCategories"(
"ParentID", "CategoryName", "IsValid", "SortWeight", "OperatorID",
"OperateTime", "Attributes")
VALUES (category."ParentID", category."CategoryName",category."IsValid", category."SortWeight", category."OperatorID", category."OperateTime",
category."Attributes")
--select currval('MerchandiseCategories_ID_seq');
returning "ID"; $BODY$
LANGUAGE sql VOLATILE
COST 100;
ALTER FUNCTION func_add_merchandisecategory("MerchandiseCategories")
OWNER TO LujieZheng;

对于返回ID网上还有一种做法是

select currval('MerchandiseCategories_ID_seq');
但是一直提示找不到该序列,我在数据库中以创建该序列,这个问题暂时还没有解决 在调用该函数的时候也出现了很多问题,因为是复合类型,所以格式为:select * from func_add_merchandisecategory(( )),需要加两个括号,括号内的复合类型的各个字段的在赋值时也有区别、对于boolean、character varying()、timestamp without time zone,text类型都要加上单引号
category.ID = await db.Database.SqlQuery<int>($"select * from func_add_merchandisecategory((0,{category.ParentID},'{category.CategoryName}','{category.IsValid}',{category.SortWeight},{category.OperatorID},'{category.OperateTime}','{{{attrs}}}'))").FirstAsync();
注:{{{}}}可以输出{}

  

上一篇:FFmpeg源代码简单分析:常见结构体的初始化和销毁(AVFormatContext,AVFrame等)


下一篇:cmd命令行模式开启或关闭 windows功能 (转载)