HOW TO: 實作 Helper Class, 以便用於 SqlParameter 的 IN 子句中的值清單

文章翻譯 文章翻譯
文章編號: 555167 - 檢視此文章適用的產品。
作者?William Ryan MVP
全部展開 | 全部摺疊

徵狀

您嘗試傳入至 SQL 陳述式或預存程序在 IN 陳述式將使用此清單分隔字元, 以分隔多個值, 而且您正在傳遞成單一清單中 SqlParameter.

發生的原因

這是因實際上的功能?ADO.NET 有益要使用在第一時間, 讓參數 - 它們逸出值, 在插入並將其傳遞中為一常值。 因此如果您設定 @ SomeValue = 1,2,3,4,5 並它用於 SELECT FROM [ TableName ] * WHERE [ ColumnName ] IN @ SomeValue, 您會不符合 [ ColumnName ] 中的值等於 1,2,3,4 或 5 所 相反地您可以只符合值, 而完全 " 1,2,3,4,5 " 因此, 如果這是我們的程序:
<CODE>
CREATE PROCEDURE up_ProcedureHelper
???? @ 語彙基元 (Token) AS VARCHAR (50) -- 或是您將需要任何長度
AS
SELECT * FROM [ TableName ]
WHERE [ FieldName ] IN @ 語彙基元 (Token))
</CODE>
這段程式碼將會失敗:
<CODE>
字串 cs = ConfigurationSettings.AppSettings("ConnectString"));
using(SqlConnection cn = new SqlConnection(cs)){
????? SqlCommand cmd = 新 SqlCommand (" up_ProcedureHelper ", ConnectionName) ;
????? cmd.CommandType = CommandType . StoredProcedure;
????? cmd.Parameters.Add("@token", "1,2,3,4,5);
????? ; cn.Open () 和 ExecuteReader Connection.Open //In 實際上應該包裝在 Try / Catch / Finally
????? dr = cmd.ExecuteReader(CommandBehavior.CloseConnection);????
????? Debug . Assert (dr.HasRows, " 我們應該有資料列但不要 」) ; //Fails 因為 ColumnName 是一個 Integer 欄位, 將不會保留 「 1,2,3,4,5 "

}

解決方案

變更您的預存程序至 Accommodate 新方法
? 如果要解決這個問題容易一種是若要建立暫存資料表使用 SQL Server 2000 的暫存資料表功能。 我們使用簡單的迴圈 (Loop), 以剖析透過我們已傳入, 以指定的分隔符號分隔的變數。 在每個回合執行迴圈, 我們會將每個變數插入暫存資料表。 最後, 我們會使用子查詢來符合 LIKE 子句中的值:
<CODE>
CREATE PROCEDURE up_ProcedureHelper
???? @ 語彙基元 (Token) AS VARCHAR (50) -- 或是您將需要任何長度
AS
DECLARE @ AS VARCHAR(50) 暫存
CREATE TABLE # HOLDER(TokenValues VARCHAR(50))
WHILE LEN(@tokens) > 0
??? BEGIN
?????? IF CHARINDEX(',', @tokens) = 0
????????? BEGIN
???????????? SET @TEMP = @tokens??
???????????? SET @tokens = ''
???????????? INSERT INTO VALUES(@Temp) #Holder (TokenValues)
????????? END 鍵
?????? ELSE
????????? BEGIN
???????????? SET @temp = 向左 (@tokens, CHARINDEX(',', @tokens) - 1)?????
???????????? INSERT INTO VALUES(@Temp) #Holder (TokenValues)
???????????? SET @tokens = RIGHT (@tokens, LEN(@tokens)-LEN(@Temp)-1)
?????? END 鍵
??? END 鍵
SELECT * FROM [ TableName ]
WHERE [ FieldName ] IN (<SELECT TokenValues FROM #holder)
</CODE>
?
所以, 假設參數 1,2,3,4,5 ", " 我們會有 [ Temp ] 資料表名稱分別為 #Holder 具有 5 列和 1,2,3,4 & 5 中的值。 只要是現場連線, 其下建立這個資料表將會存在。 只要我們關閉資料表會移除自動。 在一般情況下就絕對必須, 記得關閉您的 connections. 您不要, 以取得任何機會在這裡並希望取得連線關閉, 是使用?finally在您的程序和呼叫區塊?.Close() 方法從該處。 如果您使用 C# , 您可以在使用您的區塊包裝陳述式配合您的?finally 區塊並確保?SqlConnection是?closed 以及?disposed 的相同原則 (只在所有其他連線類型)
<CODE>
字串 cs = ConfigurationSettings.AppSettings("ConnectString"));
using(SqlConnection cn = new SqlConnection(cs)){
????? SqlCommand cmd = 新 SqlCommand (" up_ProcedureHelper ", ConnectionName) ;
????? cmd.CommandType = CommandType . StoredProcedure;
????? cmd.Parameters.Add("@token", "1,2,3,4,5");
????? ; cn.Open () 和 ExecuteReader Connection.Open //In 實際上應該包裝在 Try / Catch / Finally
????? dr = cmd.ExecuteReader(CommandBehavior.CloseConnection);????
????? Debug.Assert(dr.HasRows, "We should have Rows but Don't") ; //Now 判斷提示 (Assertion) 成功 - 和我們有 5 個資料列。
}
</CODE>

其他相關資訊

Calling?SQL Server 中參數化預存程序
良好的討論 Dynamic Sql?而且當很理想解決方案
為什麼要在良好的討論區 Parameterize ?查詢
如何以程式設計方式將動態 SQL 陳述式, Parameterized?查詢

屬性

文章編號: 555167 - 上次校閱: 2004年7月24日 - 版次: 1.0
這篇文章中的資訊適用於:
  • Microsoft ADO.NET 1.1
  • Microsoft ADO.NET 1.0
關鍵字:?
kbpubtypecca kbpubmvp kbhowto KB555167 KbMtzh kbmt
機器翻譯
重要:本文是以 Microsoft 機器翻譯軟體翻譯而成,而非使用人工翻譯而成。Microsoft 同時提供使用者人工翻譯及機器翻譯兩個版本的文章,讓使用者可以依其使用語言使用知識庫中的所有文章。但是,機器翻譯的文章可能不盡完美。這些文章中也可能出現拼字、語意或文法上的錯誤,就像外國人在使用本國語言時可能發生的錯誤。Microsoft 不為內容的翻譯錯誤或客戶對該內容的使用所產生的任何錯誤或損害負責。Microsoft也同時將不斷地就機器翻譯軟體進行更新。如果您發現錯誤,並想要協助我們進行改善,請填寫本篇文章下方的問卷。
按一下這裡查看此文章的英文版本:555167
社區解決方案內容免責聲明
Microsoft及(或)其供應商不就任何在本伺服器上發表的文字資料及其相關圖表資訊的恰當性作任何承諾。所有文字資料及其相關圖表均以「現狀」供應,不負任何擔保責任。Microsoft及(或)其供應商謹此聲明,不負任何對與此資訊有關之擔保責任,包括關於適售性、適用於某一特定用途、權利或不侵權的明示或默示擔保責任。Microsoft及(或)其供應商無論如何不對因或與使用本伺服器上資訊或與資訊的實行有關而引起的契約、過失或其他侵權行為之訴訟中的特別的、間接的、衍生性的損害或任何因使用而喪失所導致的之損害、資料或利潤負任何責任。

提供意見

 

Contact us for more help

Contact us for more help
Connect with Answer Desk for expert help.
Get more support from smallbusiness.support.microsoft.com