SQL中為什麼不要使用1=1?

来源:https://www.cnblogs.com/bossma/p/18024285
-Advertisement-
Play Games

最近看幾個老項目的SQL條件中使用了1=1,想想自己也曾經這樣寫過,略有感觸,特別拿出來說道說道。編寫SQL語句就像炒菜,每一種調料的使用都會影響菜品的最終味道,每一個SQL條件的加入也會影響查詢的執行效率。那麼 1=1 存在什麼樣的問題呢?為什麼又會使用呢? ...


最近看幾個老項目的SQL條件中使用了1=1,想想自己也曾經這樣寫過,略有感觸,特別拿出來說道說道。

編寫SQL語句就像炒菜,每一種調料的使用都會影響菜品的最終味道,每一個SQL條件的加入也會影響查詢的執行效率。那麼 1=1 存在什麼樣的問題呢?為什麼又會使用呢?

為什麼會使用 1=1?

在動態構建SQL查詢時,開發者可能會不確定最終需要哪些條件。這時候,他們就會使用“1=1”作為一個始終為真的條件,讓接下來的所有條件都可以方便地用“AND”連接起來,就像是搭積木的時候先放一個基座,其他的積木塊都可以在這個基座上疊加。

就像下邊這樣:

SELECT * FROM table WHERE 1=1
<if test="username != null">
    AND username = #{username}
</if>
<if test="age > 0">
    AND age = #{age}
</if>

這樣就不用在增加每個條件之前先判斷是否需要添加“AND”。

1=1 帶來的問題

性能問題

我們先來瞭解一下資料庫查詢優化器的工作原理。查詢優化器就像是一個聰明的圖書管理員,它知道如何最快地找到你需要的書籍。當你告訴它所需書籍的特征時,它會根據這些信息選擇最快的檢索路徑。比如你要查詢作者是“譚浩強”的書籍,它就選擇先通過作者索引找到書籍索引,再通過書籍索引找到對應的書籍,而不是費力的把所有的書籍遍歷一遍。

但是,如果我們告訴它一些無關緊要的信息,比如“我要一本書,它是一本書”,這並不會幫助管理員更快地找到書,反而可能會讓他覺得困惑。一個帶有“1=1”的查詢可能會讓資料庫去檢查每一條記錄是否滿足這個始終為真的條件,這就像是圖書管理員不得不檢查每一本書來確認它們都是書一樣,顯然是一種浪費。

不過這實際上可能也不會產生問題,因為現代資料庫的查詢優化器已經非常智能,它們通常能夠識別出像 1=1 這樣的恆真條件,併在執行查詢計劃時優化掉它們。在許多情況下,即使查詢中包含了1=1,資料庫的性能也不會受到太大影響,優化器會在實際執行查詢時將其忽略。

代碼質量

不過,我們仍然需要避免在查詢中包含 1=1,有以下幾點考慮:

  1. 代碼清晰性:即使資料庫可以優化掉這樣的條件,但對於閱讀SQL代碼的人來說,1=1可能會造成困惑。代碼的可讀性和清晰性非常重要,特別是在團隊協作的環境中。
  2. 習慣養成:即使在當前的資料庫系統中1=1不會帶來性能問題,習慣了寫不必要的代碼可能會在其他情況下引入實際的性能問題。比如,更複雜的無用條件可能不會那麼容易被優化掉。
  3. 優化器的限制:雖然現代優化器很強大,但它們並不是萬能的。在某些複雜的查詢場景中,即使是簡單的 1=1 也可能對優化器的決策造成不必要的影響,比如索引的使用。
  4. 跨資料庫相容性:不同的資料庫管理系統(DBMS)可能有不同的優化器能力。一個系統可能輕鬆優化掉1=1,而另一個系統則可能不那麼高效。編寫不依賴於特定優化器行為的SQL語句是一個好習慣。

編寫儘可能高效、清晰和準確的SQL語句,不僅有助於保持代碼的質量,也讓代碼具有更好的可維護性和可擴展性。

替代 1=1 的更佳做法

現在開發者普遍使用ORM框架來操作資料庫了,還在完全手寫拼SQL的同學可能需要反思下了,這裡給兩個不同ORM框架下替代1=1的方法。

假設我們有一個用戶信息表 user,並希望根據傳入的參數動態地過濾用戶。

首先是Mybatis

<!-- MyBatis映射文件片段 -->
<select id="selectUsersByConditions" parameterType="map" resultType="com.example.User">
  SELECT * FROM user
  <where>
    <!-- 使用if標簽動態添加條件 -->
    <if test="username != null and username != ''">
      AND username = #{username}
    </if>
    <if test="age > 0">
      AND age = #{age}
    </if>
    <!-- 更多條件... -->
  </where>
</select>

在 MyBatis 中,避免使用 WHERE 1=1 的典型方法是利用動態SQL標簽(如 <if>)來構建條件查詢。<where> 標簽會自動處理首條條件前的 AND 或 OR。當沒有滿足條件的 <if> 或其他條件標簽時,<where> 標簽內部的所有內容將被忽略,從而不會生成多餘的 AND 或 WHERE 子句。

再看看 Entity Framework 的方法:

var query = context.User.AsQueryable();
if (!string.IsNullOrEmpty(username))
{
    query = query.Where(b => b.UserName.Contains(username));
}
if (age>0)
{
    query = query.Where(b => b.Age = age);
}
var users = query.ToList();

這是一種函數式編程的寫法,最終生成SQL時,框架會決定是否在條件前增加AND,而不需要人為的增加 1=1。

總結

“1=1”在SQL語句中可能看起來無害,但實際上它是一種不良的編程習慣,可能會導致性能下降。就像在做飯時不會無緣無故地多加調料一樣,我們在編寫SQL語句時也應該避免添加無意義的條件。

每一行代碼都應該有它存在的理由,不要讓你的資料庫像一個困惑的圖書管理員,浪費時間在不必要的事情上。

  • 本文作者: 螢火架構
  • 本文鏈接: https://www.cnblogs.com/bossma/p/18024285
  • 關於博主: 使用微信掃描左側二維碼關註我的訂閱號,每天獲取新知識
  • 版權聲明: 本博客所有文章除特別聲明外,均採用 BY-NC-SA 許可協議。轉載請註明出處!

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

    -Advertisement-
    Play Games
    更多相關文章
    • 虛擬線程(Virtual Threads)是 Java 21 所有新特性中最為吸引人的內容,它可以大大來簡化和增強Java應用的併發性。但是,隨著這些變化而來的是如何最好地管理此吞吐量的問題。本文,就讓我們看一下開發人員在使用虛擬線程時,應該如何管理吞吐量。 在大多數情況下,開發人員不需要自己創建虛 ...
    • 首先,跨域的域是什麼? 跨域的英文是:Cross-Origin。 Origin 中文含義為:起源,源頭,出生地。 在跨域中,"域"指的是一個 Web 資源(比如網頁、腳本、圖片等)的源頭。 包括該資源的協議、主機名、埠號。 在同源策略中,如果兩個資源的域相同,則它們屬於同一域,可以自由進行交互和共 ...
    • 通過使用Python編程語言,編寫腳本來自動化Excel和CSV之間的轉換過程,可以批量處理大量文件,定期更新數據,並集成轉換過程到自動化工作流程中。本文將介紹如何使用第三方庫Spire.XLS for Python 實現: 使用Python將Excel轉為CSV 使用Python 將CSV轉為Ex ...
    • 多年不用PageHelper了,最近新入職的公司,採用了此工具集成的框架,作為一個獨立緊急項目開發的基礎。項目開發起來,還是手到擒來的,但是沒想到,最終測試的時候,深深的給我上了一課。 我的項目發生了哪些奇葩現象? 一切的問題都要從我接受的項目開始說起, 在開發這個項目的過程中,發生了各種奇葩的事情 ...
    • OOM 幾乎是筆者工作中遇到的線上 bug 中最常見的,一旦平時正常的頁面線上上出現頁面崩潰或者服務無法調用,查看伺服器日誌後你很可能會看到“Caused by: java.lang.OutOfMlemoryError: Java heap space” 這樣的提示,那麼毫無疑問表示的是 Java ... ...
    • 引入依賴 <dependency> <groupId>com.alibaba.csp</groupId> <artifactId>sentinel-core</artifactId> <version>1.8.7</version> </dependency> 基本用法 try (Entry ent ...
    • 一、程式計數器 程式計數器記憶體很小,可以看作是當前線程所執行位元組碼的行號指示器。 有了它,程式就能被正確的執行。 因為有線程切換的存在,則每個線程必須有各自獨立的程式計數器,即線程私有的記憶體。 這裡再解釋一下什麼是線程切換,線程切換指的是: 單處理器在執行多線程時所進行的線程切換,多線程的交替運行會 ...
    • 常見內置序列類型(Sequence Type) 類型 英文名 對應關鍵字 構造函數 是否可變 列表 list list list() 可變 元組 tuple tuple tuple() 不可變 數字序列:range range range range() 不可變 文本序列:字元串 string st ...
    一周排行
      -Advertisement-
      Play Games
    • 下麵是一個標準的IDistributedCache用例: public class SomeService(IDistributedCache cache) { public async Task<SomeInformation> GetSomeInformationAsync (string na ...
    • 這個庫提供了在啟動期間實例化已註冊的單例,而不是在首次使用它時實例化。 單例通常在首次使用時創建,這可能會導致響應傳入請求的延遲高於平時。在註冊時創建實例有助於防止第一次Request請求的SLA 以往我們要在註冊的時候實例單例可能會這樣寫: //註冊: services.AddSingleton< ...
    • 最近公司的很多項目都要改單點登錄了,不過大部分都還沒敲定,目前立刻要做的就只有一個比較老的項目 先改一個試試手,主要目標就是最短最快實現功能 首先因為要保留原登錄方式,所以頁面上的改動就是在原來登錄頁面下加一個SSO登錄入口 用超鏈接寫的入口,頁面改造後如下圖: 其中超鏈接的 href="Staff ...
    • Like運算符很好用,特別是它所提供的其中*、?這兩種通配符,在Windows文件系統和各類項目中運用非常廣泛。 但Like運算符僅在VB中支持,在C#中,如何實現呢? 以下是關於LikeString的四種實現方式,其中第四種為Regex正則表達式實現,且在.NET Standard 2.0及以上平... ...
    • 一:背景 1. 講故事 前些天有位朋友找到我,說他們的程式記憶體會偶發性暴漲,自己分析了下是非托管記憶體問題,讓我幫忙看下怎麼回事?哈哈,看到這個dump我還是非常有興趣的,居然還有這種游戲幣自助機類型的程式,下次去大玩家看看他們出幣的機器後端是不是C#寫的?由於dump是linux上的程式,剛好win ...
    • 前言 大家好,我是老馬。很高興遇到你。 我們為 java 開發者實現了 java 版本的 nginx https://github.com/houbb/nginx4j 如果你想知道 servlet 如何處理的,可以參考我的另一個項目: 手寫從零實現簡易版 tomcat minicat 手寫 ngin ...
    • 上一次的介紹,主要圍繞如何統一去捕獲異常,以及為每一種異常添加自己的Mapper實現,並且我們知道,當在ExceptionMapper中返回非200的Response,不支持application/json的響應類型,而是寫死的text/plain類型。 Filter為二方包異常手動捕獲 參考:ht ...
    • 大家好,我是R哥。 今天分享一個爽飛了的面試輔導 case: 這個杭州兄弟空窗期 1 個月+,面試了 6 家公司 0 Offer,不知道問題出在哪,難道是杭州的 IT 崩盤了麽? 報名面試輔導後,經過一個多月的輔導打磨,現在成功入職某上市公司,漲薪 30%+,955 工作制,不咋加班,還不捲。 其他 ...
    • 引入依賴 <!--Freemarker wls--> <dependency> <groupId>org.freemarker</groupId> <artifactId>freemarker</artifactId> <version>2.3.30</version> </dependency> ...
    • 你應如何運行程式 互動式命令模式 開始一個互動式會話 一般是在操作系統命令行下輸入python,且不帶任何參數 系統路徑 如果沒有設置系統的PATH環境變數來包括Python的安裝路徑,可能需要機器上Python可執行文件的完整路徑來代替python 運行的位置:代碼位置 不要輸入的內容:提示符和註 ...