淺析SQL查詢語句未顯式指定排序方式,無法保證同樣的查詢每次排序結果都一致的原因

来源:http://www.cnblogs.com/wy123/archive/2016/12/17/6189100.html
-Advertisement-
Play Games

本文出處:http://www.cnblogs.com/wy123/p/6189100.html 標題有點拗口, 先拋出問題:一個查詢沒有明確指定排序方式,那麼,第二次執行這個同樣的查詢的時候,查詢結果會不會與第一次的查詢結果排序方式完全一樣? 答案是不確定的,兩個完全一樣的查詢,結果也完全一樣,兩 ...


  

本文出處:http://www.cnblogs.com/wy123/p/6189100.html 

 

 

  標題有點拗口,
  先拋出問題:一個查詢沒有明確指定排序方式,那麼,第二次執行這個同樣的查詢的時候,查詢結果會不會與第一次的查詢結果排序方式完全一樣?
  答案是不確定的,兩個完全一樣的查詢,結果也完全一樣,兩次(多次)查詢結果的排序方式有可能一致,有可能不一致。
  如果不一致,又是什麼原因導致同樣的查詢預設排序方式不一致?
  以下簡單分析幾種情況,說明為什麼查詢同樣的查詢會出現預設排序結果不一樣的情況。當然對於該問題,包含但不限於以下幾種情況。

 

場景1:並行查詢導致預設結果集的排序是隨機的

按照慣例,先造一個表供測試

create table TestDefaultOrder1
(
    id int identity(1,1) primary key,
    col2 varchar(50),
    col3 varchar(50),
    col4 varchar(50),
    col5 varchar(50),
    col6 varchar(50),
    col7 varchar(50),
    col8 varchar(50),
    CreateDate Datetime
)
go

declare @i int =0 
begin tran
    while @i<500000
    begin
        insert into TestDefaultOrder1 values (NEWID(),NEWID(),NEWID(),NEWID(),NEWID(),NEWID(),NEWID(),GETDATE()-RAND()*500)
        set @i=@i+1
    end
commit

 

測試場景:

  這裡先不考慮索引之類的性能問題,
  如圖是一個測試結果的示例,可以看到,兩個查詢的條件是完全一樣的,都沒有顯式指定排序列,預設結果的排序是完全不一樣的

    

  甚至可以用同樣的條件做三次查詢(可以更多次),結果依然都是完全不一致的

  

 

 

原因分析:

  為什麼一樣的查詢,每次查詢結果的排序都不一樣,正如上面所說,這種情況下是並行查詢導致的。
  查詢引擎採用什麼樣的執行計劃是基於代價考慮的,如果一旦發現一個查詢的執行代價超過一定的閾值,就有可能採用並行的方式來處理,
  如果採用了並行查詢的方式,就會採用多個線程來分解整個查詢任務,而每一個線程分配的任務量是無法固定的,同時,合併每個線程的結果順序也是不固定的
  這就導致了最終的查詢結果的順序是不固定的。
  截圖即為並行查詢的每個線程分配的任務量示例。

  如圖,當前這個查詢,第一個線程返回的行數是2,但是無法保證第二次查詢的第一個線程返回的行數也是2,
  即便是第二次返回的行數是2,也無法保證返回的2行與第一次返回的兩行數據一樣的
  同時,在合併各個線程的結果集的時候,依據線程返回的時間來的,理論上講也是不確定的,多個不確定因素在一起,就造成了最終的結果集排序(可以認為)是隨機的。

  

 

場景2:物理存儲導致預設結果的隨機性

  同樣,先造一個測試數據的case,如下,創建一個堆表,

create table TestDefaultOrder2
(
    id int identity(1,1),
    col2 char(5000)
)
go

declare @i int =0 
begin tran
    while @i<50
    begin
        insert into TestDefaultOrder2 values (NEWID())
        set @i=@i+1
    end
commit

 

測試場景:

  這個場景排除了上述並行查詢的影響,因為只有50條數據,根本不會啟用並行查詢
  如截圖,兩次查詢之間執行了一次表的重建動作,同樣是數據本身沒有發生任何變化,兩次查詢的預設順序完全不一樣

 

甚至在重建一次,查詢結果仍然與上面兩次還是都不一樣的。

 

 原因分析:

  堆表的特點決定了堆內的數據行和數據頁沒有任何固定的順序,整個堆內的數據在物理存儲發生了變化之後,
  在對查詢(對堆表掃描)的過程中得不到一個與物理存儲變化之前完全一樣的順序。
  除了上述的重建表會導致查詢的預設順序不一致,其他影響物理空間的操作,都會影響堆表數據頁面的物理存儲位置,

  比如這裡再執行一次資料庫的收縮,收縮之後的查詢與收縮之前的查詢順序依舊是不一樣的,我可沒有動你表和你表中的任何一條數據,但你不能阻止我正常的資料庫維護操作。
  總之,一旦影響到物理存儲位置,堆表的預設掃描結果順序都有可能不一樣。

  

 

  以上僅僅通過單表查詢來說明,如果未顯式指定排序方式,即便是同樣的查詢條件,查詢結果的順序是無法保證每次都一致的,
  如果是多表關聯,或者是考慮到索引,資料庫維護等操作,情況將變得更加複雜,比如這個也比較有意思:http://www.cnblogs.com/wy123/p/5425946.html
  比較特殊的是:沒有顯式指定排序方式,
    1,某段一個時間段內,查詢結果可能是按照預期結果排序的,某個時間段內就不是了(物理存儲改變的影響);
    2,某些查詢條件下是按照預期結果排序的,改變一下查詢條件,排序結果就變得面目全非了(執行計劃改變的影響)。
  總之一句話:沒有顯式執行排序方式,不要期待查詢結果每次都是預期的排序方式,甚至每次都不一樣。

 

 

 總結:

   本文通過兩個簡單的示例,
  從執行計劃和物理存儲兩個方面,說明瞭“如果查詢SQL沒有顯式指定排序方式,查詢結果的順序是無法保證總是按照你的預期來的”。
  當然也不能局限於這兩種情況。
  然而話不能說死,某些條件下沒有顯式指定排序方式,可能會得到預期的排序結果,但是這種期待往往是不可靠的。
  

  “昨天系統查詢結果的排序還是好好的,今天怎麼變了?”
  “為啥我用A條件查詢是按照時間排序的,按照B條件查詢就不是了?”
  如果沒有顯式指定排序方式,不要問我資料庫是不是有問題(或者說SQL Server這個資料庫“不行”,或者說DBA說是內部原因是忽悠人的)。


  所以同學,如果期望查詢結果排序,不管預設是不是你預期的排序方式,都請顯式指定排序方式。

 

 

 

 


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

-Advertisement-
Play Games
更多相關文章
  • 設計思路:給UIView增加一個分類 所有的視圖都可以根據需要來進行紅點顯示 ...
  • 在OC當中自動計算行高主要調用系統的 - (--boundingRectWithSize:(CGSize)size options:(NSStringDrawingOptions)options attributes:(nullable NSDictionary<NSString *, id> *) ...
  • GridView 可以指定顯示的條目的列數。 listview一般顯示的條目的列數都是一列 如果是列表(單列多行形式)的使用ListView,如果是多行多列網狀形式的優先使用GridView android:numColumns=”auto_fit” //GridView的列數設置為自動 GridV ...
  • 現在二維碼的應用越來越普及,二維碼掃描也成為手機應用程式的必備功能了。本文將基於 Xamarin.Android 平臺使用 ZXing.Net.Mobile 做一個簡單的 Android 條碼掃描示常式序。 1、新建一個 Xamarin.Android 應用: 閱讀全文 ...
  • 苦心人天不負, 為了項目終於把 iOS 10 跳轉系統設置的方法給搞定了, 很欣慰. http://www.cnblogs.com/lurenq/p/6189580.html iOS 10 跳轉系統設置的欄位 電池電量 Prefs:root=BATTERY_USAGE 通用設置 Prefs:root ...
  • 從今天開始研究開發自己的編程語言Ocelot,從《自製編譯器》出發,然後再自己不斷完善功能並優化。 編譯器前端簡單,就不深入研究了,直接用現成的一款工具叫JavaCC,它可以生成抽象語法樹,抽象語法樹是生成中間代碼的關鍵,而中間代碼又是生成後端代碼的關鍵。 整個編譯器代碼採用java語言編寫,主要功 ...
  • 本文用於收集在運維中經常使用的系統內置函數,持續整理中 一,常用Metadata函數 1,查看資料庫的ID和Name 2,查看對象的ID和Name,對象的Schema,對象的定義 3,查看Schema的ID和Name,通過對象ID獲取對象的架構名(Schema) 4,查看Column的Name 二, ...
  • 基本概念: 資料庫DB(database): + 數據的倉庫,數據的集合,是數據的一種結構化的存儲 資料庫管理系統DBMS(database management system): + 管理資料庫的一套軟體 + 比如Oracle、MySQL、SQL server、DB2 + 介於應用程式和操作系統之 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...