所以我有一个查询需要从外部应用程序获取输入并使用这些值插入表变量。
DECLARE @prompt TABLE(
answerID int,
rowid int NOT NULL IDENTITY(1,1) primary key
);
INSERT INTO @HELLO (answerid)
SELECT (x,y,z)
问题在于,当收到输入时,它会以如下列表的形式出现:
INSERT INTO @HELLO (answerid)
SELECT (4,55,66,88,978)
哪种语法不正确。该列表明确地以x,y,z的形式出现,因此我没有办法按摩数据,如:
INSERT INTO #blah
VALUES (x), (y), (z)
还是有吗?最后,我只需要将这个值列表放到表中,这样我就可以对它们进行处理了。
答案 0 :(得分:2)
如果列表是从C#等传入的,并且源自集合(比如DataTable),那么处理此列表的最有效方法是使用表值参数。首先,创建一个表类型:
CREATE TYPE dbo.MyList(ID INT PRIMARY KEY);
现在,创建一个接受此类型作为参数的存储过程:
CREATE PROCEDURE dbo.MyProcedure
@List dbo.MyList READONLY
AS
BEGIN
SET NOCOUNT ON;
DECLARE @prompt TABLE
(
answerID INT,
[rowid] INT NOT NULL IDENTITY(1,1) PRIMARY KEY
);
INSERT INTO @HELLO (answerid)
SELECT ID FROM @List;
...
END
GO
T-SQL的示例用法:
DECLARE @List dbo.MyList;
INSERT @List VALUES(4),(55),(66),(88),(978);
EXEC dbo.MyProcedure @List = @List;
从C#等,您可以直接传递DataTable作为参数,这样您就不必担心所有这些转置。
如果你不能解决这个问题,那就创建一个Split函数。
CREATE FUNCTION dbo.SplitInts
(
@List VARCHAR(MAX),
@Delimiter VARCHAR(255) = ','
)
RETURNS TABLE
WITH SCHEMABINDING
AS
RETURN
(
SELECT Item = y.i.value('(./text())[1]', 'int')
FROM
(
SELECT x = CONVERT(XML, '<i>'
+ REPLACE(@List, @Delimiter, '</i><i>')
+ '</i>').query('.')
) AS a CROSS APPLY x.nodes('i') AS y(i)
);
GO
现在你可以说:
DECLARE @prompt TABLE
(
answerID INT,
[rowid] INT NOT NULL IDENTITY(1,1) PRIMARY KEY
);
INSERT @prompt(answerID)
SELECT Item FROM dbo.SplitInts('4,55,66,88,978', ',');