хранимая процедура с переменным количеством параметров - PullRequest
16 голосов
/ 18 сентября 2011

У меня есть хранимая процедура, в которой я должен передать параметры, но проблема в том, что я не уверен, сколько параметров придет, это может быть 1, при следующем запуске это может быть 5.

cmd.Parameters.Add(new SqlParameter("@id", id)

Может кто-нибудь помочь, как я могу передать эти переменные количество параметров в хранимой процедуре?Спасибо

Ответы [ 6 ]

20 голосов
/ 19 сентября 2011

Вы можете передать его в виде списка через запятую, затем использовать функцию разделения и объединить результаты.

CREATE FUNCTION dbo.SplitInts
(
   @List       VARCHAR(MAX),
   @Delimiter  CHAR(1)
)
RETURNS TABLE
AS
   RETURN 
   (
       SELECT Item = CONVERT(INT, Item)
       FROM
       (
           SELECT Item = x.i.value('(./text())[1]', 'INT')
           FROM
           (
               SELECT [XML] = CONVERT(XML, '<i>' 
                    + REPLACE(@List, @Delimiter, '</i><i>') 
                    + '</i>').query('.')
           ) AS a
           CROSS APPLY
           [XML].nodes('i') AS x(i)
       ) AS y
       WHERE Item IS NOT NULL
   );

Теперь ваша хранимая процедура:

CREATE PROCEDURE dbo.doStuff
    @List VARCHAR(MAX)
AS
BEGIN
    SET NOCOUNT ON;

    SELECT cols FROM dbo.table AS t
        INNER JOIN dbo.SplitInts(@List, ',') AS list
        ON t.ID = list.Item;
END
GO

Тогда, чтобы назвать это:

EXEC dbo.doStuff @List = '1, 2, 3, ...';

Здесь вы можете увидеть некоторые фоны, другие параметры и сравнения производительности:

В SQL Server 2016 или более поздней версии вы должны посмотреть STRING_SPLIT() и STRING_AGG():

6 голосов
/ 18 сентября 2011

Хранимые процедуры поддерживают необязательные параметры.Как и в C # 4, вы можете указать значение по умолчанию, используя =.Например:

create procedure dbo.doStuff(
     @stuffId int = null, 
     @stuffSubId int = null, 
     ...)
as
...

Для параметров, которые вы не хотите передавать, либо установите их на null, либо вообще не добавляйте их в cmd.Parameters.У них будет значение по умолчанию в хранимой процедуре

5 голосов
/ 18 сентября 2011

SQLServer позволяет передавать параметр TABLE в хранимую процедуру.Таким образом, вы можете определить тип таблицы, CREATE TYPE LIST_OF_IDS AS TABLE (id int not null primary key), изменить вашу процедуру так, чтобы она принимала переменную этого типа (она должна быть доступна только для чтения).

2 голосов
/ 18 сентября 2011

Рассматривали ли вы использовать словарь для этой цели? Это позволит вам передавать любое количество параметров в виде пар ключ-значение. Затем вам нужно просто просмотреть словарь и добавить эти параметры в cmd.

void DoStuff(Dictionary<string, object> parameters)
{
    // some code
    foreach(var param in parameters)
    {
        cmd.Parameters.Add(new SqlParameter(param.Key, param.Value);
    }
    // some code
}

В самой хранимой процедуре вам необходимо указать значения по умолчанию для параметров.

CREATE PROCEDURE DoStuff(
     @id INT = NULL,
     @value INT = NULL,
     -- the list of parameters with their default values goes here
     )
AS
-- procedure body
0 голосов
/ 06 декабря 2018

Если у вас Sql Server 2008 или более поздняя версия, вы можете использовать параметр с табличным значением ...

https://blog.sqlauthority.com/2008/08/31/sql-server-table-valued-parameters-in-sql-server-2008/

0 голосов
/ 27 марта 2018

Вот фрагмент кода, который разбивает строку на основе , в качестве разделителя.Вы даже можете параметризовать запятую.Это полезно в системах, которые еще не имеют функции String_split:

  DECLARE @startindex INT
  DECLARE @commaindex INT
  DECLARE @paramAsString VARCHAR(MAX) -- this represents the input param
  DECLARE @param VARCHAR (1024)
  DECLARE @paramsTable TABLE(param VARCHAR(1024) NOT NULL) -- the splitted params come here

  SET @startindex = 1
  WHILE @startindex < LEN(@paramAsString)
  BEGIN
    SET @commaindex = CHARINDEX(',', @paramAsString, @startindex)

    IF @commaindex = 0
    BEGIN
      SET @param = SUBSTRING(@paramAsString, @startindex, LEN(@paramAsString))
      SET @startindex = LEN(@settlementEntities)
    END
    ELSE
    BEGIN
      SET @param = SUBSTRING(@paramAsString, @startindex, (@commaindex - @startindex))
      SET @startindex = @commaindex + 1
    END
    IF @se IS NOT NULL AND 0 < LEN(RTRIM(LTRIM(@param)))
    BEGIN
      SET @param = RTRIM(LTRIM(@param))
      INSERT INTO @paramsTable (param) VALUES (@param)
    END
  END
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...