C# 使用 SqlParameter 加入查詢條件

December 25, 2023

C# 中有 SqlCommand 類別可以放入 Raw SQL,查詢時可以使用 SqlParamter 類別加入查詢的條件,並避免 SQL Injection 導致的資安風險。

使用方式

SQL 查詢語法中,為要加入的查詢條件設定名稱,並在前方加上 @ 符號。

SqlCommand 類別設定查詢語法,並在 Parameters.Add 方法傳入 SqlParameter。第一個參數為查詢條件的名稱,第二個參數為查詢的變數。

若傳入的變數內容包含 SQL Injection 的控制符號時,會拋出例外。

以下是範例:

SqlCommand command = new SqlCommand("SELECT * FROM Dogs1 WHERE Name LIKE @Name", connection);
command.Parameters.Add(new SqlParameter("Name", dogName));

另外,因為 SqlParameter 實作 DbParameter 抽象類別,當方法需要傳入 DbParameter 時,可以直接建立 SqlParameter。從以下範例中,可看到建立 SqlParameter 時直接指定屬性名稱、值。

var parameters = new[]{
            new SqlParameter(){ ParameterName="foo", Value="hello" },
            new SqlParameter(){ ParameterName="bar", Value="World" }
        };
x.ExecuteScalar<int>(commandText, commandType, parameters);

如何在 IN 條件加入參數

如果今天要查詢的 SQL 語法,裡面的條件包含用 IN 查詢多個條件的話,可以自己組多個參數和查詢語法。

StackOverflow 上的解法如下:

string[] values = new [] {"value1", "value2", "value3", "value4"};
  StringBuilder query = new StringBuilder("Select * From Table Where Column in (");
  SqlCommand cmd = new SqlCommand();
  cmd.Connection = new SqlConnection("Your connection string");
  for(int i = 0; i < columns.Length; i++)
  {
      string arg = string.Format("@arg{0}", i);
      cmd.Parameters.AddwithValue(arg, SanatizeSqlString(columns[i]));
      sb.AppendFormat("{0}, ", arg);
  }
  sb = sb.Remove(sb.Length -2, 2);
  sb.Append(")");
  cmd.CommandText = sb.ToString();

最後的查詢語法呈現如下:

Select * From Table Where Column in (@arg0, @arg1, @arg2, @arg3)

正確的處理資料

如果資料庫欄位允許插入 Null,那麼可以用以下方式指定 Null 值:

decimal? price = 100m;
var parameter = new SqlParameter { ParameterName = $"price", Value = (object)price ?? DBNull.Value}

另外,建議指定插入的資料類型,避免系統判斷失誤,而導致存放不正確的資料,或執行時出現轉型的錯誤訊息,例如:

decimal? price = 100m;
var parameter = new SqlParameter { ParameterName = $"price", Value = (object)price ?? DBNull.Value, DbType = System.Data.DbType.Decimal}

參考資料