使用 ODBC 資料表值參數
本主題將討論搭配 ODBC 使用資料表值參數的主要使用者案例:
完整繫結多資料列緩衝區之下的資料表值參數 (使用記憶體中的所有值,將資料當做 TVP 傳送)
以資料流方式傳送資料列的資料表值參數 (使用資料執行中,將資料當做 TVP 傳送)
從系統目錄擷取資料表值參數中繼資料
針對準備好的陳述式擷取資料表值參數中繼資料
完整繫結多資料列緩衝區之下的資料表值參數 (使用記憶體中的所有值,將資料當做 TVP 傳送)
當搭配完整繫結的多資料列緩衝區使用時,記憶體中將提供所有參數值。 例如,這對於 OLTP 交易而言就是典型的情況,在這類交易中,資料表值參數可以封裝到單一預存程序中。 如果沒有資料表值參數,這會牽涉到動態產生複雜的多重陳述式批次,或是對伺服器進行多次呼叫。
資料表值參數本身會使用 SQLBindParameter 與其他參數系結。 系結所有參數之後,應用程式會在每個資料表值參數上設定參數焦點屬性SQL_SOPT_SS_PARAM_FOCUS,並針對資料表值參數的資料行呼叫 SQLBindParameter。
資料表值參數的伺服器類型是新的SQL Server特定類型,SQL_SS_TABLE。 SQL_SS_TABLE 的繫結 C 類型一定必須是 SQL_C_DEFAULT。 資料表值參數繫結的參數不會傳送任何資料;它是用來傳遞資料表中繼資料,以及控制要如何傳遞資料表值參數之組成資料行中的資料。
資料表值參數的長度會設定為傳送給伺服器的資料列數。 資料表值參數 SQLBindParameter 的 ColumnSize 參數會指定可傳送的資料列數目上限;這是資料行緩衝區的陣列大小。 ParameterValuePtr 是 SQLBindParameter 中資料表值參數的參數緩衝區。 ParameterValuePtr 及其相關聯的 BufferLength 會在需要時傳遞資料表值參數的類型名稱。 此類型名稱不是預存程序呼叫所需,但為 SQL 陳述式所需。
在對 SQLBindParameter 的呼叫上指定資料表值參數類型名稱時,它一律必須指定為 Unicode 值,即使在建置為 ANSI 應用程式的應用程式中也一樣。 當您使用 SQLSetDescField 指定資料表值參數類型名稱時,您可以使用符合應用程式建置方式的常值。 ODBC 驅動程式管理員將會執行所有必要的 Unicode 轉換。
資料表值參數和資料表值參數資料行的中繼資料可透過使用 SQLGetDescRec、SQLSetDescRec、SQLGetDescField 和 SQLSetDescField 來個別和明確操作。 不過,多載 SQLBindParameter 通常更方便,而且在大部分情況下不需要明確的描述元存取。 此方法與其他資料類型的 SQLBindParameter 定義一致,不同之處在于,對於資料表值參數,受影響的描述元欄位稍有不同。
有時應用程式會搭配動態 SQL 使用資料表值參數,而且必須提供此資料表值參數的類型名稱。 如果這是這種情況,而且資料表值參數未定義于連線的目前預設架構中,則必須使用 SQLSetDescField 來設定SQL_CA_SS_TYPE_CATALOG_NAME和SQL_CA_SS_TYPE_SCHEMA_NAME。 因為資料表類型定義和資料表值參數必須在相同的資料庫中,所以如果應用程式使用資料表值參數,就不能設定 SQL_CA_SS_TYPE_CATALOG_NAME。 否則,SQLSetDescField 會報告錯誤。
此案例的範例程式碼位於使用 Table-Valued Parameters (ODBC) 中的程式 demo_fixed_TVP_binding
。
以資料流方式傳送資料列的資料表值參數 (使用資料執行中,將資料當做 TVP 傳送)
在此案例中,應用程式會在要求時提供資料列給驅動程式,然後以資料流方式將資料列傳送給伺服器。 如此就不需要在記憶體中緩衝處理所有資料列。 這是大量插入/更新案例的代表。 資料表值參數會提供參數陣列與大量複製之間某一處的效能點。 也就是說,編寫資料表值參數就跟參數陣列一樣輕鬆,但是資料表值參數在伺服器上提供更大的彈性。
如同上一節「完整繫結多資料列緩衝區之下的資料表值參數」所討論,資料表值參數和它的資料行會繫結在一起,但是資料表值參數本身的長度指標會設定為 SQL_DATA_AT_EXEC。 驅動程式會傳回SQL_NEED_DATA,以執行中資料參數的一般方式回應 SQLExecute 或 SQLExecuteDirect。 當驅動程式準備好接受資料表值參數的資料時,SQLParamData 會傳回 SQLBindParameter 中的 ParameterValuePtr 值。
應用程式會針對資料表值參數使用 SQLPutData 來指出資料表值參數組成資料行的資料可用性。 針對資料表值參數呼叫 SQLPutData 時, DataPtr 必須一律為 null, 而且StrLen_or_Ind 必須是小於或等於資料表值參數緩衝區所指定的陣列大小, (SQLBindParameter) 的 ColumnSize 參數。 0 表示資料表值參數沒有其他資料列,而且此驅動程式將會繼續處理下一個實際程序參數。 當 StrLen_or_Ind 不是 0 時,驅動程式會以與非資料表值參數系結參數相同的方式處理資料表值參數組成資料行:每個資料表值參數資料行都可以指定其實際資料長度、SQL_Null_DATA,也可以透過其長度/指標緩衝區指定資料。 資料表值參數資料行值可以在字元或二進位值傳入片段時,重複呼叫 SQLPutData 來傳遞。
當所有資料表值參數資料行都已經處理過之後,此驅動程式會回到資料表值參數來進一步處理資料表值參數資料的資料列。 因此,如果是資料執行中的資料表值參數,此驅動程式不會遵循一般的繫結參數循序掃描。 系結資料表值參數將會輪詢,直到呼叫 SQLPutData 且StrLen_Or_IndPtr 等於 0,此時驅動程式會略過資料表值參數資料行,並移至下一個實際的預存程式參數。 當 SQLPutData 傳遞大於或等於 1 的指標值時,驅動程式會循序處理資料表值參數資料行和資料列,直到其具有所有系結資料列和資料行的值為止。 然後此驅動程式會回到資料表值參數。 從 SQLParamData 接收資料表值參數的權杖,以及呼叫 SQLPutData (hstmt、Null、n) 資料表值參數之間,應用程式必須設定資料表值參數組成資料行資料,以及要傳遞至伺服器之下一個資料列或資料列的指標緩衝區內容。
此案例的範例程式碼位於使用 Table-Valued Parameters (ODBC) 中的常式 demo_variable_TVP_binding
。
從系統目錄擷取資料表值參數中繼資料
當應用程式針對具有資料表值參數參數的程式呼叫 SQLProcedureColumns 時,DATA_TYPE會以SQL_SS_TABLE傳回,而 TYPE_NAME 是資料表值參數的資料表類型名稱。 將兩個額外的資料行新增至 SQLProcedureColumns 所傳回的結果集:SS_TYPE_CATALOG_NAME傳回目錄的名稱,其中已定義資料表值參數的資料表類型,而SS_TYPE_SCHEMA_NAME傳回架構的名稱,其中定義了 table-value 參數的資料表類型。 與 ODBC 規格一致,SS_TYPE_CATALOG_NAME和SS_TYPE_SCHEMA_NAME會出現在舊版SQL Server中新增的所有驅動程式特定資料行之前,以及 ODBC 本身所授權的所有資料行之後。
新的資料行不但會針對資料表值參數擴展,也會針對 CLR 使用者定義型別參數擴展。 仍然會擴展 UDT 參數現有的結構描述和目錄資料行,但是讓需要的資料類型擁有共同的結構描述和目錄資料行將會簡化未來的應用程式開發過程 (請注意,XML 結構描述集合會有些不同,而且未包含在這項變更中)。
應用程式會使用 SQLTable 來判斷資料表類型的名稱,與永續性資料表、系統資料表和檢視表的名稱相同。 新的資料表類型 TABLE TYPE 已經導入,可讓應用程式識別與資料表值參數相關聯的資料表類型。 資料表類型和一般表格會使用不同的命名空間。 這表示,您可以將相同的名稱用於資料表類型和實際資料表。 為了處理這個情況,已經導入了新的陳述式屬性 SQL_SOPT_SS_NAME_SCOPE。 這個屬性會指定採用資料表名稱做為參數的 SQLTable 和其他目錄函式,是否應該將資料表名稱解譯為實際資料表的名稱或資料表類型的名稱。
應用程式會使用 SQLColumns,以永續性資料表的相同方式來判斷資料表類型的資料行,但必須先設定SQL_SOPT_SS_NAME_SCOPE,以指出它正在使用資料表類型,而不是實際資料表。 SQLPrimaryKeys 也可以搭配資料表類型使用,再次使用 SQL_SOPT_SS_NAME_SCOPE。
此案例的範例程式碼位於使用 Table-Valued Parameters (ODBC) 中的常式 demo_metadata_from_catalog_APIs
。
針對準備好的陳述式擷取資料表值參數中繼資料
在此案例中,應用程式會使用 SQLNumParameters 和 SQLDescribeParam 來擷取資料表值參數的中繼資料。
IPD 欄位 SQL_CA_SS_TYPE_NAME 是用來擷取資料表值參數的類型名稱。 IPD 欄位 SQL_CA_SS_TYPE_SCHEMA_NAME 和 SQL_CA_SS_TYPE_CATALOG_NAME 是分別用來擷取它的目錄和結構描述。
資料表類型定義和資料表值參數必須在相同的資料庫中。 如果應用程式在使用資料表值參數時設定SQL_CA_SS_TYPE_CATALOG_NAME,SQLSetDescField 將會報告錯誤。
SQL_CA_SS_TYPE_CATALOG_NAME 和 SQL_CA_SS_TYPE_SCHEMA_NAME 也可以用來擷取與 CLR 使用者定義型別參數相關聯的目錄和結構描述。 SQL_CA_SS_TYPE_CATALOG_NAME 和 SQL_CA_SS_TYPE_SCHEMA_NAME 是 CLR UDT 類型的現有類型特有之目錄結構描述屬性的替代方式。
應用程式也會使用 SQLColumns 在此案例中擷取資料表值參數的資料行中繼資料,因為 SQLDescribeParam 不會傳回資料表值參數資料行之資料行的中繼資料。
此使用案例的範例程式碼位於使用 Table-Valued Parameters (ODBC) 中的常式 demo_metadata_from_prepared_statement
。