Sqlserver
 sql >> база данни >  >> RDS >> Sqlserver

Как да предадете масив в съхранена процедура на SQL Server

SQL Server 2008 (или по-нова версия)

Първо, във вашата база данни създайте следните два обекта:

CREATE TYPE dbo.IDList
AS TABLE
(
  ID INT
);
GO

CREATE PROCEDURE dbo.DoSomethingWithEmployees
  @List AS dbo.IDList READONLY
AS
BEGIN
  SET NOCOUNT ON;

  SELECT ID FROM @List; 
END
GO

Сега във вашия C# код:

// Obtain your list of ids to send, this is just an example call to a helper utility function
int[] employeeIds = GetEmployeeIds();

DataTable tvp = new DataTable();
tvp.Columns.Add(new DataColumn("ID", typeof(int)));

// populate DataTable from your List here
foreach(var id in employeeIds)
    tvp.Rows.Add(id);

using (conn)
{
    SqlCommand cmd = new SqlCommand("dbo.DoSomethingWithEmployees", conn);
    cmd.CommandType = CommandType.StoredProcedure;
    SqlParameter tvparam = cmd.Parameters.AddWithValue("@List", tvp);
    // these next lines are important to map the C# DataTable object to the correct SQL User Defined Type
    tvparam.SqlDbType = SqlDbType.Structured;
    tvparam.TypeName = "dbo.IDList";
    // execute query, consume results, etc. here
}

SQL Server 2005

Ако използвате SQL Server 2005, все пак бих препоръчал функция за разделяне на XML. Първо, създайте функция:

CREATE FUNCTION dbo.SplitInts
(
   @List      VARCHAR(MAX),
   @Delimiter VARCHAR(255)
)
RETURNS TABLE
AS
  RETURN ( SELECT Item = CONVERT(INT, Item) FROM
      ( SELECT Item = x.i.value('(./text())[1]', 'varchar(max)')
        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
  );
GO

Сега вашата съхранена процедура може да бъде просто:

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

  SELECT EmployeeID = Item FROM dbo.SplitInts(@List, ','); 
END
GO

И във вашия C# код просто трябва да предадете списъка като '1,2,3,12' ...

Намирам, че методът за преминаване през параметри с таблица опростява поддръжката на решение, което го използва и често има повишена производителност в сравнение с други реализации, включително XML и разделяне на низове.

Входните данни са ясно дефинирани (никой не трябва да гадае дали разделителят е запетая или точка и запетая) и нямаме зависимости от други функции за обработка, които не са очевидни без проверка на кода за съхранената процедура.

В сравнение с решения, включващи дефинирана от потребителя XML схема вместо UDT, това включва подобен брой стъпки, но според моя опит кодът е много по-прост за управление, поддръжка и четене.

В много решения може да се нуждаете само от един или няколко от тези UDT (Потребителски дефинирани типове), които използвате повторно за много съхранени процедури. Както в този пример, общото изискване е да се премине през списък с указатели на ID, името на функцията описва какъв контекст трябва да представляват тези идентификатори, името на типа трябва да е общо.



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Прозрачно криптиране на данни (TDE) в SQL Server в група за наличност на AlwaysOn в пример

  2. Връщане на множество таблици от съхранена процедура

  3. Как да създадете база данни в SQL Server

  4. Какво е функция с таблично значение в SQL Server?

  5. Как да върна множество набори от резултати със SqlCommand?