資料表值參數 (Database Engine)
資料表值參數是 SQL Server 2008 中的新參數類型。資料表值參數是使用使用者定義的資料表類型所宣告。您可以使用資料表值參數,將多個資料列傳送到 Transact-SQL 陳述式或常式 (如預存程序或函數),而不需要建立暫存資料表或許多參數。
資料表值參數就像是 OLE DB 和 ODBC 中的參數陣列,但是提供了更多的彈性,而且與 Transact-SQL 更緊密整合在一起。資料表值參數也會因為能夠參與以集合為基礎的作業而獲益。
[!附註]
Transact-SQL 會以傳址方式將資料表值參數傳遞給常式,以免產生輸入資料的複本。
您可以使用資料表值參數來建立及執行 Transact-SQL 常式,然後從 Transact-SQL 程式碼 (任何 Managed 語言中的 Managed 和原生用戶端) 呼叫這些常式。
在 Transact-SQL 中建立及使用資料表值參數
資料表值參數有兩個主要元件:SQL Server 類型以及參考該類型的參數。若要建立及使用資料表值參數,請遵循以下步驟:
建立資料表類型及定義資料表結構。
如需有關如何建立 SQL Server 類型的詳細資訊,請參閱<使用者定義資料表類型>。如需有關如何定義資料表結構的詳細資訊,請參閱<CREATE TABLE (Transact-SQL)>。
宣告一個常式,此常式具有資料表類型的參數。如需有關 SQL Server 常式的詳細資訊,請參閱<CREATE PROCEDURE (Transact-SQL)>和<CREATE FUNCTION (Transact-SQL)>。
宣告資料表類型的變數,並參考此資料表類型。如需有關如何宣告變數的詳細資訊,請參閱<DECLARE @local\_variable (Transact-SQL)>。
請使用 INSERT 陳述式填入此資料表變數。如需有關如何插入資料的詳細資訊,請參閱<使用 INSERT 與 SELECT 加入資料列>。
當建立及填入此資料表變數之後,您可以將此變數傳遞給常式。
如果此常式超出範圍,就不能再使用資料表值參數。類型定義在被捨棄之前仍然可用。
若要在 SQL Server Native Client 中使用資料表值參數,請參閱<資料表值參數 (SQL Server Native Client)>。
若要在 ADO.NET 中使用資料表值參數,請參閱 ADO.NET 文件集。
優點
資料表值參數提供更大的彈性,而且在某些情況下,其效能優於暫存資料表或是傳遞參數清單的其他方法。資料表值參數提供下列好處:
不需要從用戶端鎖定初始擴展的資料。
提供簡單的程式設計模型。
可讓您將複雜的商務邏輯併入單一常式內。
減少與伺服器之間的往返次數。
可以有一個不同基數的資料表結構。
具有強型別。
可讓用戶端指定排序次序和唯一索引鍵。
限制
資料表值參數有下列限制:
SQL Server 不會維護資料表值參數之資料行上的統計資料。
資料表值參數必須當做輸入 READONLY 參數傳遞給 Transact-SQL 常式。您不能在常式主體內針對資料表值參數執行 DML 作業,例如 UPDATE、DELETE 或 INSERT。
您不能使用資料表值參數當做 SELECT INTO 或 INSERT EXEC 陳述式的目標。資料表值參數可以在 SELECT INTO 的 FROM 子句中或是 INSERT EXEC 字串或預存程序內。
範圍
資料表值參數的範圍為預存程序、函數或動態 Transact-SQL 文字,與其他參數一模一樣。同樣地,資料表類型之變數的範圍與使用 DECLARE 陳述式建立的其他任何區域變數一樣。您可以在動態 Transact-SQL 陳述式內宣告資料表值變數,並將這些變數當做資料表值參數傳遞給預存程序和函數。
安全性
資料表值參數的權限遵循 SQL Server 的物件安全性模型,其方式是使用 Transact-SQL 關鍵字:CREATE、GRANT、DENY、ALTER、CONTROL、TAKE OWNERSHIP、REFERENCES、EXECUTE、VIEW DEFINITION 和 REVOKE。
目錄檢視
若要取得與資料表值參數有關的資訊,您可以查詢下列目錄檢視:sys.parameters (Transact-SQL)、sys.types (Transact-SQL) 和 sys.table_types (Transact-SQL)。
資料表值參數與 BULK INSERT 作業的比較
使用資料表值參數可以和使用以集合為基礎之變數的其他方式相比較;但是,對於大型資料集而言,使用資料表值參數通常可以更快速。與大量作業 (其啟動成本高於資料表值參數) 相較之下,當插入 1000 個資料列以下時,資料表值參數的執行效能會很不錯。
重複使用的資料表值參數會因為暫存資料表快取而獲益。這種資料表快取提供了比同等的 BULK INSERT 作業更好的延展性。藉由使用小型資料列插入作業,可能會因為使用參數清單或批次處理的陳述式 (而非 BULK INSERT 作業或資料表值參數) 而獲得一些效能上的好處。但是,這些方法的便利性不如程式,而且當資料列增加時,效能會快速降低。
資料表值參數的執行效能等於或優於同等的參數陣列實作。
下表將根據插入作業的速度顯示所要使用的技術。
資料來源 |
伺服器邏輯 |
資料列數目 |
最佳技術 |
---|---|---|---|
伺服器上格式化的資料檔 |
直接插入 |
< 1000 |
BULK INSERT |
伺服器上格式化的資料檔 |
直接插入 |
> 1000 |
BULK INSERT |
伺服器上格式化的資料檔 |
複雜 |
< 1000 |
資料表值參數 |
伺服器上格式化的資料檔 |
複雜 |
> 1000 |
BULK INSERT |
遠端用戶端處理序 |
直接插入 |
< 1000 |
資料表值參數 |
遠端用戶端處理序 |
直接插入 |
> 1000 |
BULK INSERT |
遠端用戶端處理序 |
複雜 |
< 1000 |
資料表值參數 |
遠端用戶端處理序 |
複雜 |
> 1000 |
資料表值參數 |
範例
下列範例使用 Transact-SQL 並示範如何建立資料表值參數類型、宣告變數來參考它、填入參數清單,然後將值傳遞給預存程序。
USE AdventureWorks;
GO
/* Create a table type. */
CREATE TYPE LocationTableType AS TABLE
( LocationName VARCHAR(50)
, CostRate INT );
GO
/* Create a procedure to receive data for the table-valued parameter. */
CREATE PROCEDURE usp_InsertProductionLocation
@TVP LocationTableType READONLY
AS
SET NOCOUNT ON
INSERT INTO [AdventureWorks].[Production].[Location]
([Name]
,[CostRate]
,[Availability]
,[ModifiedDate])
SELECT *, 0, GETDATE()
FROM @TVP;
GO
/* Declare a variable that references the type. */
DECLARE @LocationTVP
AS LocationTableType;
/* Add data to the table variable. */
INSERT INTO @LocationTVP (LocationName, CostRate)
SELECT [Name], 0.00
FROM
[AdventureWorks].[Person].[StateProvince];
/* Pass the table variable data to a stored procedure. */
EXEC usp_InsertProductionLocation @LocationTVP;
GO