今天是周日,刚好有空闲时间整理一下这些天工作业务中遇到的问题。

有时候我们有这样一个需求,就是在后台中传过来一个IList<类>的泛型集合数据,该集合是某个类的实例集合体,然后将该集合中的实例的数据一个个地插入到数据库或者更新到数据库中去。一开始我想到的方法是拼接字符串,然后通过存储过程对接收到的字符串进行截取,再一个个地插入或者更新到数据库中去,这是最原始的方法,不过过程会比较复杂,想到这就头疼。后来查找发现说SqlServer2008中为存储过程添加了一个新特性,可以传递表类型的参数,既然可以传递表类型参数,那问题就变得简单啦。以下是开发中写的code.

1.asp.net后台:

         /// <summary>
/// Add the PayrollCycle
/// </summary>
/// <param name="payrollCycle">payrollCycle</param>
/// <returns>bool</returns>
public bool AddPayrollCycle(IList<PayrollCycle> payrollCycles)
{
DataTable dataTable=new DataTable();
dataTable.Columns.Add("Year",typeof(int));
dataTable.Columns.Add("Month",typeof(int));
dataTable.Columns.Add("CutoffDate",typeof(int));
dataTable.Columns.Add("PayrollDate",typeof(int));
dataTable.Columns.Add("EnterUser",typeof(string));
dataTable.Columns.Add("EnterDate",typeof(DateTime));
dataTable.Columns.Add("LastUpdatedUser",typeof(string));
dataTable.Columns.Add("LastUpdatedDate",typeof(DateTime));
foreach (PayrollCycle p in payrollCycles)
{
DataRow dataRow = dataTable.NewRow();
dataRow["Year"] = p.Year;
dataRow["Month"] = p.Month;
dataRow["CutoffDate"] = p.CutoffDate;
dataRow["PayrollDate"] = p.PayrollDate;
dataRow["EnterUser"] = UserSession.LogOnUserAccount;
dataRow["EnterDate"] = DateTime.Now;
dataRow["LastUpdatedUser"] = UserSession.LogOnUserAccount;
dataRow["LastUpdatedDate"] = DateTime.Now;
dataTable.Rows.Add(dataRow);
} SqlParameter[]paras=new SqlParameter[]
{
new SqlParameter("@PayrollCycles",dataTable)
};
return SqlHelper.ExecuteNonQuery("MCU.USP_AddPayrollCycles", paras) > ;
}

为dataTable添加column的时候,必须明确该列的typeof,否则在存储过程当中会把传入的该列当成varchar类型看待,导致某些类型转换失败

2.在SqlServer中先定义一个Table类型的Type:

 CREATE TYPE [PayrollCycleType] AS TABLE(
[YEAR] [int] NOT NULL,
[Month] [int] NOT NULL,
[CutoffDate] [int] NOT NULL,
[PayrollDate] [int] NOT NULL,
[EnterUser] [varchar]() NULL,
[EnterDate] [datetime] NULL,
[LastUpdatedUser] [varchar]() NULL,
[LastUpdatedDate] [datetime] NULL
)
GO

接着编写一个传入上步骤中定义的表类型的参数的存储过程,该参数为Readonly(作为表类型参数必须为可读),代码如下:

插入操作:

 CREATE PROCEDURE [MCU].[USP_AddPayrollCycles]
(
@PayrollCycles MCU.PayrollCycleType Readonly
)
AS
BEGIN
SET NOCOUNT ON
BEGIN TRANSACTION
INSERT INTO MCU.PayrollCycle
( [Year] ,
[Month] ,
CutoffDate ,
PayrollDate ,
EnterUser ,
EnterDate ,
LastUpdatedUser ,
LastUpdatedDate
)
SELECT [Year] ,
[Month] ,
CutoffDate ,
PayrollDate ,
EnterUser ,
EnterDate ,
LastUpdatedUser ,
LastUpdatedDate
FROM @PayrollCycles
COMMIT TRANSACTION
END GO

更新操作:

 CREATE PROCEDURE [MCU].[USP_UpdatePayrollCycle]
(
@PayrollCycles MCU.PayrollCycleType READONLY,
@TypeOfDate NVARCHAR()
)
AS
BEGIN
SET NOCOUNT ON
--declare an table
DECLARE @temp AS MCU.PayrollCycleType
--insert into @temp from @PayrollCycles
INSERT INTO @temp
( [YEAR] ,
[Month] ,
CutoffDate ,
PayrollDate
)
SELECT
[YEAR] ,
[Month] ,
CutoffDate ,
PayrollDate
FROM @PayrollCycles --Update the PayrollCycle
IF(@TypeOfDate='Payroll')
BEGIN
UPDATE MCU.PayrollCycle
SET
PayrollDate=t.PayrollDate
FROM
@temp t
WHERE MCU.PayrollCycle.[Year]=t.[Year] AND MCU.PayrollCycle.[Month]=t.[Month]
END IF(@TypeOfDate='Cut-off')
BEGIN
UPDATE MCU.PayrollCycle
SET
CutoffDate=t.CutOffDate
FROM @temp t
WHERE MCU.PayrollCycle.[Year]=t.[Year] AND MCU.PayrollCycle.[Month]=t.[Month]
END
END GO

就这样大功告成,用起来相当方便,就不用通过拼接字符串进行数据的插入,更新操作了。

SqlServer存储过程传入Table参数的更多相关文章

  1. PROCEDURE存储过程传入表参数

    ) ,itemNum ) ,itemQty )) ---2.创建一个存储过程以表值参数作为输入 alter proc usp_TestProcWithTable     @tb  LocationTa ...

  2. clob字段的值插入和查询N种方法【包括java调用存储过程传入clob参数】

    import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.IOException; import jav ...

  3. SQL 存储过程 传入数组参数

    今天在做统计数据的时候,传入数组导致数据不显示.解决方式和大家分享一下: --参数@CompanyName='北京,天津,上海' DECLARE @PointerPrev int     DECLAR ...

  4. SQLSERVER 根据传入的参数拼接sql语句字符串,反馈结果集

    ALTER PROCEDURE [dbo].[usp_visit_detail](@siteid BIGINT, @Startime VARCHAR(15), @Endtime  VARCHAR(15 ...

  5. php通过存储过程传入汉字参数并写入数据库

    写入数据库,汉字为???样式的乱码,后根据网上介绍的方法,参数前加N,数据库汉字内容变成空白. 解决方法,在PHP中先转为base64,再在mysql中base64解码,前提先保证mysql中有bas ...

  6. sqlserver 存储过程 带输出参数

    CREATE PROCEDURE [dbo].[output] @acctNbr varchar(), --会员卡号 @acctPwd1 nvarchar() OUT, --登录密码 @acctPwd ...

  7. SQLSERVER procedure 传入参数为DataTable类型 C#该怎么写

    以上为数据库中存储过程传入参数为table类型 table类型在数据库中存在为: 最后在C#实现方式为:

  8. SQLSERVER存储过程语法详解

    CREATE PROC [ EDURE ] procedure_name [ ; number ] [ { @parameter data_type } [ VARYING ] [ = default ...

  9. 创建并在项目中调用SQLSERVER存储过程的简单示例

    使用SQLSERVER存储过程可以很大的提高程序运行速度,简化编程维护难度,现已得到广泛应用.创建存储过程 和数据表一样,在使用之前需要创建存储过程,它的简明语法是: 引用: Create PROC ...

随机推荐

  1. Enable-Migrations 在应用程序配置文件中找不到xx连接字符串

    在解决方案中有多个项目时,使用Enable-Migrations 命令进行数据迁移时,出现以下错误: 尝试在Enable-Migrations 命令中指定-projectName也不行,最后将要操作的 ...

  2. Java中的Clone机制(浅层复制)

    浅层复制代码: import java.util.*; class Int{ private int i; public Int(int ii){i = ii;} public void increm ...

  3. js单例模式

    js实现单例模式,经常使用两种方法,一种是使用构造函数的静态属性中缓存该实例,另一种是将实例包装在闭包中. 第一种实现方式: //静态属性中单例模式 function Universe() { if ...

  4. 从配置文件中读取数据获取Connection

    配置文件 db.driver=com.mysql.jdbc.Driver db.url=jdbc\:mysql\://localhost\:3306/mybase db.user=root db.ps ...

  5. 通过jqueryui实现邮件提示

    //js代码$(function () { var availableTags = ["@qq.com", "@gmail.com", "@126.c ...

  6. python学习第二天 --变量及其字符串

    python变量: 在计算机程序中,变量不仅可以是数字,还可以是任意数据类型. 在Python程序中,变量是用一个变量名表示,变量名必须是大小写英文.数字和下划线(_)的组合,且不能用数字开头. 在P ...

  7. 【笔记】shell下的主要工具

    shell 下有很多工具是非常好的辅助. 一.自我精进的途径 1.help / --help :简要的帮助说明: help 是针对 shell 内建功能的帮助, gnu工具一般为 "程序名 ...

  8. C语言编程时常犯十八个错误

    C语言的最大特点是:功能强.使用方便灵活.C编译的程序对语法检查并不象其它高级语言那么严格,这就给编程人员留下“灵活的余地”,但还是由于这个灵活给程序的调试带来了许多不便,尤其对初学C语言的人来说,经 ...

  9. ORA-19573: cannot obtain exclusive enqueue for datafile 1

    还原Oracle数据库时出现ORA-19870和ORA-19573错误,如: RMAN> restore database; Starting restore at 11-DEC-12 usin ...

  10. #Java编程题-百钱百鸡

    问题: 百钱百鸡问题.用100钱买100只鸡,公鸡一只五钱,母鸡一只三钱,雏鸡三只一钱,编程计算共有几种买法(要求每种鸡至少要买1只). 自己的实现,没有什么数据结构,算法,求大神指点!! packa ...