關於EF中直接執行sql語句的參數化問題

来源:http://www.cnblogs.com/xie-zhonglai/archive/2017/05/25/EF_SqlParamter.html
-Advertisement-
Play Games

某天 , 在review項目中代碼的時候, 發現有哥們直接通過 Database.ExecuteSqlCommand("select * from order_info where company like '%abc%' ")的方式與資料庫查詢, 私下問其是否知道這個方法還有一個帶有params ...


某天 , 在review項目中代碼的時候, 發現有哥們直接通過 Database.ExecuteSqlCommand("select * from order_info where  company like '%abc%' ")的方式與資料庫查詢, 私下問其是否知道這個方法還有一個帶有params object[] parameters參數重載, 對方爽快的回答:"知道啊 ,只是每次都要寫一大堆的SqlParameter[] 定義, 實在是太過麻煩了,特別是有的時候查詢參數太多了...." .細想一下, 確實麻煩得很,感覺又回到了直接寫sqlhelper的時代了.不過轉念一想, 這個方法很容易改進, 於是便有了這篇隨筆,希望能給有需要的朋友一些提示.

1.目標,我們希望調用者能像字元串格式化,類似   string.Formart("select  * from order_info where company like {0}","abc") 一樣去方便使用,而不再關心是否是否需要參數化,但同時資料庫在執行的時候,自動轉化為參數化的語句.

2.略加思考:首先我們利用 string.Formart 自動將sqlCommand 格式化為帶參數的一個字元串, 類似 "select  * from order_info where company like  @p0", 同時自動創建查詢參數@p0= abc,即可.

3.於是,便有瞭如下方法片段

private Tuple<string, SqlParameter[]> ProcessSqlCommand(string sqlCommand, params object[] param)
{
   var tempParamKeyValDic = param.Select((item, i) => new KeyValuePair<string, object>("@p" + i, item))
                               .ToDictionary(k => k.Key, v => v.Value); 
            
   var tempSqlCommand = string.Format(sqlCommand, tempParamKeyValDic.Keys.ToArray());
   var tempParams = tempParamKeyValDic.Select(t => new SqlParameter(t.Key, t.Value)).ToArray();

   return Tuple.Create(tempSqlCommand, tempParams);
}

 

4. 同時對外提供兩個類似的ExecSqlCommand 和 SqlQuery 方法即可.

public int ExecSqlCommand(string sqlCommand, params object[] param)
{
    if (param == null || param.Length == 0)
    {
        return Context.Database.ExecuteSqlCommand(sqlCommand);
    }

    var temp = ProcessSqlCommand(sqlCommand, param);

    return Context.Database.ExecuteSqlCommand(temp.Item1, temp.Item2);
}

public List<T> SqlQuery<T>(string sqlCommand, params object[] param)
{
    if (param == null || param.Length == 0)
    {
        return Context.Database.SqlQuery<T>(sqlCommand).ToList();
    }

    var temp = ProcessSqlCommand(sqlCommand, param);

    return Context.Database.SqlQuery<T>(temp.Item1, temp.Item2).ToList();
}

 

5. 客戶端在調用的時候,直接使用類似如下模式

 var data=repository.SqlQuery<OrderInfo>(“select  * from order_info where company like  {0}","abc”);

 

後記:

      1. 事關數據安全無小事, 任何僥幸心理都應該避免;

      2. 當有人在埋怨程式不好寫, 或者框架不好用的時候 , 需要換位思考,並極力改進;

      3. 大家在談論項目框架或系統架構的時候,並不是時時刻刻都需要關註類似高大上的高併發架構圖, 有的時候還得壓住浮躁的心態關註眼前的蹩腳之處.


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、將解壓後的“KindEditor”文件夾複製到項目的根目錄(此KindEditor文件夾已經經過優化) 下載地址為: http://pan.baidu.com/s/1eS1PRii 2、在<head>標簽中引入KindEditor外部樣式表文件和必需的外部js文件。如下所示: <!--引入樣式表 ...
  • Parallel類是對線程的一個抽象。該類位於System.Threading.Tasks名稱空間中,提供了數據和任務並行性。 Paraller類定義了數據並行地For和ForEach的靜態方法,以及任務並行的Invoke的靜態方法。Parallel.For()和Parallel.ForEach() ...
  • 類似相關問題有以下: WCF- restful介面 POST方式調用報錯(遠程伺服器返回錯誤: 400 錯誤的請求) WCF Rest:不使用UriTemplate使用post方式傳參解決HTTP400問題以及參數映射問題 等等! 具體原因參照:原創:轉載請標明出處:http://www.cnblo ...
  • 一、新建一個html頁面,如註冊頁面"Register.htm" 二、新建一js文件,如:reg.js 三、處理ajax請求 方法一:手動拼接json字元串 新建一般處理程式,如:Register.ashx 方法二:使用Json.NET工具來將C#對象轉換json輸出 1、新建信息類“Msg.cs” ...
  • 前言 很多時候其實我們並不需要asp.net core自帶的那麼複雜的用戶系統,基於角色,各種概念,還得用EF Core,而且在web應用中都是把信息存儲到cookie中進行通訊(我不喜歡放cookie中,因為有次我在mac系統中的safari瀏覽器運行web應用時,碰到跨域cookie設不上,非要 ...
  • 回到目錄 TransactionScope是.net環境下的事務,可以提升為分散式事務,這些知識早在很久前就已經說過了,今天不再說它,今天主要談談Savechanges()這個方法在TransactionScope塊里的作用,我們知識TransactionScope只有顯示的提交動作而沒有回滾,那麼 ...
  • 首先並不是每個事件的e參數都有上述兩個屬性。 e.Cancel:獲取或設置指示是否應取消事件的值;e.Handled:獲取或設置一個值,該值指示是否處理過此事件。 下麵說說比較常見的場景: 1)e.cancel: ①視窗關閉,比如用戶點擊視窗右上角想關閉,但代碼里彈出確認框讓用戶確認是否真的想退出, ...
  • 1、Messager交互結構和消息類型 銜接上篇,Messeger是信使的意思,顧名思義,他的目是用於View和ViewModel 以及 ViewModel和ViewModel 之間的消息通知和接收。 Messenger類用於應用程式的通信,接受者只能接受註冊的消息類型,另外目標類型可以被指定,用S ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...