sql >> Databáze >  >> RDS >> Sqlserver

Jak předat pole do uložené procedury serveru SQL Server

SQL Server 2008 (nebo novější)

Nejprve ve své databázi vytvořte následující dva objekty:

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

Nyní ve vašem kódu 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

Pokud používáte SQL Server 2005, stále bych doporučil funkci rozdělení přes XML. Nejprve vytvořte funkci:

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

Nyní může být vaše uložená procedura:

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

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

A v kódu C# stačí předat seznam jako '1,2,3,12' ...

Zjistil jsem, že metoda předávání parametrů s hodnotou tabulky zjednodušuje udržovatelnost řešení, které ji používá, a často má vyšší výkon ve srovnání s jinými implementacemi včetně XML a dělení řetězců.

Vstupy jsou jasně definované (nikdo nemusí hádat, zda je oddělovačem čárka nebo středník) a nemáme závislosti na dalších funkcích zpracování, které nejsou zřejmé bez kontroly kódu pro uloženou proceduru.

Ve srovnání s řešeními zahrnujícími uživatelsky definované schéma XML namísto UDT to zahrnuje podobný počet kroků, ale podle mých zkušeností je mnohem jednodušší kód spravovat, udržovat a číst.

V mnoha řešeních můžete potřebovat pouze jeden nebo několik těchto UDT (uživatelsky definované typy), které znovu použijete pro mnoho uložených procedur. Stejně jako v tomto příkladu je běžným požadavkem projít seznam ukazatelů ID, název funkce popisuje, jaký kontext by tato ID měla představovat, název typu by měl být obecný.



  1. Nesprávné řazení/kompletace/pořadí s mezerami v Postgresql 9.4

  2. Jak povolit protokol dotazů MySQL?

  3. Oracle PL/SQL:Export dat z tabulky do CSV

  4. Normalizujte data transakcí ze sloupců času a stavu na minuty podle hodnoty stavu