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

有时候我们有这样一个需求,就是在后台中传过来一个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. wcf wpf

    转 http://blog.csdn.net/thunder09/article/details/5792157 WPF就是所谓下一代Windows界面层技术,我觉得还有满有前途的.不过Vista发布 ...

  2. SignalR2.0开发实例之——群发消息

    一.前言 ASP .NET SignalR 是一个ASP .NET 下的类库,可以在ASP .NET 的Web项目中实现实时通信.什么是实时通信的Web呢?就是让客户端(Web页面)和服务器端可以互相 ...

  3. 1、工程log4j 配置

    ### set log levels ### log4j.rootLogger = debug,stdout,R ### 输出到控制台 ### log4j.appender.stdout = org. ...

  4. Ruby和Rails开发环境安装

    更新包管理 sudo apt-get update 安装curl sudo apt-get install curl *安装rvm via curl \curl -L https://get.rvm. ...

  5. [转]JS基础之undefined与null的区别

    在JavaScript开发中,被人问到:null与undefined到底有啥区别? 一时间不好回答,特别是undefined,因为这涉及到undefined的实现原理.于是,细想之后,写下本文,请各位 ...

  6. Ibatis 后台打印完整的sql语句

    http://blog.csdn.net/deng11342/article/details/9122015 http://www.blogjava.net/libin2722/archive/200 ...

  7. awk多个数组的使用

    #!/bin/bash awk 'BEGIN{printf "%-60s %-10s %-10s %-10s\n","url","count" ...

  8. [C入门 - 游戏编程系列] 贪吃蛇篇(五) - 蛇实现

    因为已经写了食物的实现,所以我不知道到底是该先写世界的实现还是蛇的实现.因为世界就是一个窗口,可以立刻在世界中看到食物的样子,对于大多数人来说,如果写完代码立刻就能看到效果,那就再好不过了.可是,我最 ...

  9. nginx日志管理与限速

    1.日志简介nginx日志主要有两种:访问日志和错误日志.访问日志主要记录客户端访问nginx的每一个请求,格式可以自定义:错误日志主要记录客户端访问nginx出错时的日志,格式不支持自定义.两种日志 ...

  10. Windows 1252和ISO 8859-1之间的区别(ISO 8859-1就是Latin-1,但1252与Latin1略有不同)

    2.6.5. ANSI字符编码和Windows 1252 Windows为了支持英语和西欧字符,自己设计了一个编码,对应的在Code Page号是1252,被称为Windows 1252. Windo ...