1

所以我有一个查询需要从外部应用程序获取输入并使用这些值插入到表变量中。

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)

或者有吗?最后,我只需要将这个值列表放入一个表中,这样我就可以对它们执行工作。

4

1 回答 1

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 作为参数传递,因此您不必担心所有这些转置。

如果你不能处理,那么创建一个拆分函数。

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', ',');
于 2013-07-29T19:59:46.310 回答