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
    • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...
    • 目錄前言PostgreSql安裝測試額外Nuget安裝Person.cs模擬運行Navicate連postgresql解決方案Garnet為什麼要選擇Garnet而不是RedisRedis不再開源Windows版的Redis是由微軟維護的Windows Redis版本老舊,後續可能不再更新Garne ...
    • C#TMS系統代碼-聯表報表學習 領導被裁了之後很快就有人上任了,幾乎是無縫銜接,很難讓我不想到這早就決定好了。我的職責沒有任何變化。感受下來這個系統封裝程度很高,我只要會調用方法就行。這個系統交付之後不會有太多問題,更多應該是做小需求,有大的開發任務應該也是第二期的事,嗯?怎麼感覺我變成運維了?而 ...
    • 我在隨筆《EAV模型(實體-屬性-值)的設計和低代碼的處理方案(1)》中介紹了一些基本的EAV模型設計知識和基於Winform場景下低代碼(或者說無代碼)的一些實現思路,在本篇隨筆中,我們來分析一下這種針對通用業務,且只需定義就能構建業務模塊存儲和界面的解決方案,其中的數據查詢處理的操作。 ...
    • 對某個遠程伺服器啟用和設置NTP服務(Windows系統) 打開註冊表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\W32Time\TimeProviders\NtpServer 將 Enabled 的值設置為 1,這將啟用NTP伺服器功 ...
    • title: Django信號與擴展:深入理解與實踐 date: 2024/5/15 22:40:52 updated: 2024/5/15 22:40:52 categories: 後端開發 tags: Django 信號 松耦合 觀察者 擴展 安全 性能 第一部分:Django信號基礎 Djan ...
    • 使用xadmin2遇到的問題&解決 環境配置: 使用的模塊版本: 關聯的包 Django 3.2.15 mysqlclient 2.2.4 xadmin 2.0.1 django-crispy-forms >= 1.6.0 django-import-export >= 0.5.1 django-r ...
    • 今天我打算整點兒不一樣的內容,通過之前學習的TransformerMap和LazyMap鏈,想搞點不一樣的,所以我關註了另外一條鏈DefaultedMap鏈,主要調用鏈為: 調用鏈詳細描述: ObjectInputStream.readObject() DefaultedMap.readObject ...
    • 後端應用級開發者該如何擁抱 AI GC?就是在這樣的一個大的浪潮下,我們的傳統的應用級開發者。我們該如何選擇職業或者是如何去快速轉型,跟上這樣的一個行業的一個浪潮? 0 AI金字塔模型 越往上它的整個難度就是職業機會也好,或者說是整個的這個運作也好,它的難度會越大,然後越往下機會就會越多,所以這是一 ...
    • @Autowired是Spring框架提供的註解,@Resource是Java EE 5規範提供的註解。 @Autowired預設按照類型自動裝配,而@Resource預設按照名稱自動裝配。 @Autowired支持@Qualifier註解來指定裝配哪一個具有相同類型的bean,而@Resourc... ...