c# - 存储过程中的多值日期参数?

标签 c# tsql sql-server-2008 stored-procedures parameters

我试图让一个存储过程工作,它接受日期的多值参数。这不在 SSRS 中,但我正在尝试使用与它相同的方法:

ALTER PROCEDURE spSelectPlacementData
(
    @ClientID           SMALLINT,
    @SourceFileDates    VARCHAR(MAX)
)
AS
BEGIN
    SELECT (snip)
    FROM [APS].[dbo].[Account] A
    WHERE ClientID = @ClientID
    AND A.[SourceFileDate] IN (SELECT * FROM dbo.Split(@SourceFileDates))
END

我在 SSRS 报告多值参数上将这种方法与 INT 和 VARCHAR 字段结合使用。

这是我用来连接 SourceFileDates 的代码:

    string sourceFileDates = "";

    foreach (DateTime file in job.sourceFiles)
    {
        if (file == job.sourceFiles.Last())
        {
            sourceFileDates += "'" + file.ToString("d") + "'";
        }
        else
        {
            sourceFileDates += "'" + file.ToString("d") + "', ";
        }
    }

    selectRunCommand = new SqlCommand("spSelectPlacementData", sqlConnection);
    selectRunCommand.CommandType = CommandType.StoredProcedure;
    selectRunCommand.Parameters.Add("@ClientID", SqlDbType.SmallInt);
    selectRunCommand.Parameters["@ClientID"].Value = job.clientID;
    selectRunCommand.Parameters.Add("@SourceFileDates", SqlDbType.VarChar);
    selectRunCommand.Parameters["@SourceFileDates"].Value = sourceFileDates;

使用我在网上抓取的这个 dbo.Split 函数:

/****** Object:  UserDefinedFunction [dbo].[Split]    Script Date: 09/20/2011 11:16:13 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
ALTER FUNCTION [dbo].[Split]
/* This function is used to split up multi-value parameters */
(
@ItemList VARCHAR(MAX),
@delimiter CHAR(1)
)
RETURNS @IDTable TABLE (Item VARCHAR(MAX) collate database_default )
AS
BEGIN
DECLARE @tempItemList VARCHAR(MAX)
SET @tempItemList = @ItemList

DECLARE @i INT
DECLARE @Item VARCHAR(MAX)

SET @tempItemList = REPLACE (@tempItemList, @delimiter + ' ', @delimiter)
SET @i = CHARINDEX(@delimiter, @tempItemList)

WHILE (LEN(@tempItemList) > 0)
BEGIN
IF @i = 0
SET @Item = @tempItemList
ELSE
SET @Item = LEFT(@tempItemList, @i - 1)

INSERT INTO @IDTable(Item) VALUES(@Item)

IF @i = 0
SET @tempItemList = ''
ELSE
SET @tempItemList = RIGHT(@tempItemList, LEN(@tempItemList) - @i)

SET @i = CHARINDEX(@delimiter, @tempItemList)
END
RETURN
END

我想我并不完全清楚我如何格式化参数、SSRS 如何对类似参数进行格式化(这是我尝试从代码中做的唯一一个)以及 Date 数据如何进行格式化之间的区别type 影响所需的格式。我收到“从字符串转换日期和/或时间时转换失败。”选择多个值时出错。

编辑:根据要求,foreach 循环输出示例:

'9/9/2011', '8/19/2011', '8/12/2011'

最佳答案

为什么不使用 Table-Valued parameter

在 SQL 上创建用户定义的表类型 DateTimes

create type DateTimes as table
(
    [Value] datetime
)

然后修改你的存储过程:

ALTER PROCEDURE spSelectPlacementData
(
    @ClientID           SMALLINT,
    @SourceFileDates    DateTimes readonly -- must be readonly
)

现在您可以将@SourceFileDates 视为只读表变量。

指定 SqlCommand 参数时,表值参数指定为 SqlDbType.Structured并通过 DataTableDataRowcollection .因此,您可以像这样填充它:

var sourceFileDates = new DataTable();
sourceFileDates.Columns.Add("Value", typeof(DateTime));
foreach (DateTime file in job.sourceFiles)
{
    sourceFileDates.Rows.Add(file);
}
selectRunCommand.Parameters.Add(new SqlParameter {
    ParameterName = "@SourceFileDates", 
    Value = sourceFileDates,
    SqlDbType = SqlDbType.Structured // make sure you specify structured
});

现在一切都很好并且类型正确...而且您不必进行任何字符串解析或转换。

作为旁注,您不妨继续创建 StringsIntegers 类型;您会迷上 TVP 并在各处使用它们。

关于c# - 存储过程中的多值日期参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7487624/

相关文章:

c# - 不同 Controller 中的属性路由结果为 "Multiple controller types were found that match the URL"

c# - 通过在父级上呈现新的 View Controller viewWillDisappear 被触发,我取消订阅事件

c# - 我如何全局 Hook 所有键和所有鼠标按钮?

sql - 可以使用 DISTINCT 进行分区函数 COUNT() OVER

sql - 与聚合交叉应用

sql-server - BCP 到 .csv 分隔符问题

java - MySQL Jar 可以工作,但 MSSQL 不能?

sql - 如何从 union 语句中删除重复行

C# 反射 - 加载程序集并调用方法(如果存在)

sql - 向上或向下舍入到最接近的 20