欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

MySQL中order?by排序語句的原理解析

 更新時間:2022年12月10日 15:04:05   作者:沐沐沐晨風  
這篇文章主要介紹了MySQL中order?by排序語句的原理,本文結(jié)合示例代碼給大家講解的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下

order by 是怎么工作的?

表定義

CREATE TABLE `t1` (  
	`id` int(11) NOT NULL,  
	`city` varchar(16) NOT NULL,  
	`name` varchar(16) NOT NULL,  
	`age` int(11) NOT NULL,  
	`addr` varchar(128) DEFAULT NULL,  
	PRIMARY KEY (`id`),  
	KEY `city` (`city`)) ENGINE=InnoDB;

SQL語句可以這樣寫:

select city,name,age from t1 where city='杭州' order by name limit 1000

全字段排序

用 explain 命令來看看這個語句的執(zhí)行情況。

其中Using index condition是索引下推優(yōu)化(索引下推簡介),Using filesort 表示的就是需要排序,MySQL 會給每個線程分配一塊內(nèi)存用于排序,稱為 sort_buffer。

city索引的示意圖。

從圖中可以看到,滿足 city='杭州’條件的行,是從 ID_X 到 ID_(X+N) 的這些記錄。

通常情況下,這個語句執(zhí)行流程如下所示 :

  • 初始化 sort_buffer,確定放入 name、city、age 這三個字段;
  • 從索引 city 找到第一個滿足 city='杭州’條件的主鍵 id,也就是圖中的 ID_X;
  • 到主鍵 id 索引取出整行,取 name、city、age 三個字段的值,存入 sort_buffer 中;
  • 從索引 city 取下一個記錄的主鍵 id;
  • 重復(fù)步驟 3、4 直到 city 的值不滿足查詢條件為止,對應(yīng)的主鍵 id 也就是圖中的 ID_Y;
  • 對 sort_buffer 中的數(shù)據(jù)按照字段 name 做快速排序;
  • 按照排序結(jié)果取前 1000 行返回給客戶端。

這個是它的排序過程,叫做全字段排序,執(zhí)行流程示意圖如下所示。

按 name 排序”這個動作,可能在內(nèi)存中完成,也可能需要使用外部排序,這取決于排序所需的內(nèi)存和參數(shù) sort_buffer_size。

sort_buffer_size,就是 MySQL 為排序開辟的內(nèi)存(sort_buffer)的大小。

  • 要排序的數(shù)據(jù)量小于 sort_buffer_size,排序就在內(nèi)存中完成。
  • 要排序數(shù)據(jù)量太大,內(nèi)存放不下,利用磁盤臨時文件輔助排序。

可以用下面介紹的方法,來確定一個排序語句是否使用了臨時文件。

/* 打開optimizer_trace,只對本線程有效 */
SET optimizer_trace='enabled=on'; 

/* @a保存Innodb_rows_read的初始值 */
select VARIABLE_VALUE into @a from  performance_schema.session_status where variable_name = 'Innodb_rows_read';

/* 執(zhí)行語句 */
select city, name,age from t where city='杭州' order by name limit 1000; 

/* 查看 OPTIMIZER_TRACE 輸出 */
SELECT * FROM `information_schema`.`OPTIMIZER_TRACE`\G

/* @b保存Innodb_rows_read的當前值 */
select VARIABLE_VALUE into @b from performance_schema.session_status where variable_name = 'Innodb_rows_read';

/* 計算Innodb_rows_read差值 */
select @b-@a;

這個方法是通過查看 OPTIMIZER_TRACE 的結(jié)果來確認的,你可以從 number_of_tmp_files 中看到是否使用了臨時文件。

1670570299819

number_of_tmp_files 表示的是,排序過程中使用的臨時文件數(shù)。

為什么需要12個文件呢?

外部排序一般使用歸并排序算法??梢赃@么簡單理解,MySQL 將需要排序的數(shù)據(jù)分成 12 份,每一份單獨排序(快速排序)后存在這些臨時文件中。然后把這 12 個有序文件再合并成一個有序的大文件。

小結(jié):

如果 sort_buffer_size 超過了需要排序的數(shù)據(jù)量的大小,number_of_tmp_files 就是 0,表示排序可以直接在內(nèi)存中完成。sort_buffer_size 越小,需要分成的份數(shù)越多,number_of_tmp_files 的值就越大。

rowid排序

在上面這個算法過程里面,只對原表的數(shù)據(jù)讀了一遍,剩下的操作都是在 sort_buffer 和臨時文件中執(zhí)行的。但這個算法有一個問題,就是如果查詢要返回的字段很多的話,那么 sort_buffer 里面要放的字段數(shù)太多,這樣內(nèi)存里能夠同時放下的行數(shù)很少,要分成很多個臨時文件,排序的性能會很差。

如果當行很大,這個全字段排序并不是很好。

SET max_length_for_sort_data = 16;

這個語句的意思是:如果單行太大,超過所設(shè)定的數(shù)值的時候,比如現(xiàn)在是超過16,MySQL就認為單行太大,換一種算法。

city、name、age 這三個字段的定義總長度是 36,我把 max_length_for_sort_data 設(shè)置為 16。

新的算法放入 sort_buffer 的字段,只有要排序的列(即 name 字段)和主鍵 id。

但這時,排序的結(jié)果就因為少了 city 和 age 字段的值,不能直接返回了(最后收集結(jié)果之前要回表),整個執(zhí)行流程就變成如下所示的樣子:

  • 初始化 sort_buffer,確定放入兩個字段,即 name 和 id;
  • 從索引 city 找到第一個滿足 city='杭州’條件的主鍵 id,也就是圖中的 ID_X;
  • 到主鍵 id 索引取出整行,取 name、id 這兩個字段,存入 sort_buffer 中;
  • 從索引 city 取下一個記錄的主鍵 id;
  • 重復(fù)步驟 3、4 直到不滿足 city='杭州’條件為止,也就是圖中的 ID_Y;
  • 對 sort_buffer 中的數(shù)據(jù)按照字段 name 進行排序;
  • 遍歷排序結(jié)果,取前 1000 行,并按照 id 的值回到原表中取出 city、name 和 age 三個字段返回給客戶端。

這個執(zhí)行流程的示意圖如下,叫做 rowid 排序。

對比全字段排序流程圖發(fā)現(xiàn),rowid 排序多訪問了一次表 t 的主鍵索引,就是步驟 7。

注意:最后的**“結(jié)果集”是一個邏輯概念,實際上 MySQL 服務(wù)端從排序后的 sort_buffer 中依次取出 id,然后到原表查到 city、name 和 age 這三個字段的結(jié)果,不需要在服務(wù)端再耗費內(nèi)存存儲結(jié)果,是直接返回給客戶端的**。

全字段排序和rowid排序應(yīng)該如何去選擇呢?

如果 MySQL 實在是擔心排序內(nèi)存太小,會影響排序效率,才會采用 rowid 排序算法,這樣排序過程中一次可以排序更多行,但是需要再回到原表去取數(shù)據(jù)。

如果 MySQL 認為內(nèi)存足夠大,會優(yōu)先選擇全字段排序,把需要的字段都放到 sort_buffer 中,這樣排序后就會直接從內(nèi)存里面返回查詢結(jié)果了,不用再回到原表去取數(shù)據(jù)。

這也就體現(xiàn)了 MySQL 的一個設(shè)計思想:如果內(nèi)存夠,就要多利用內(nèi)存,盡量減少磁盤訪問。對于 InnoDB 表來說,rowid 排序會要求回表多造成磁盤讀,因此不會被優(yōu)先選擇。

其實以上說的都是無序的時候,如果在條件有索引,索引中數(shù)據(jù)是有序的,省掉了上述步驟,直接在索引上找到主鍵id,然后回表找到要查找的數(shù)據(jù)直接返回給客戶端。

到此這篇關(guān)于MySQL中order by排序語句的原理的文章就介紹到這了,更多相關(guān)MySQL中order by排序語句的原理內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SQL性能優(yōu)化方法及性能測試

    SQL性能優(yōu)化方法及性能測試

    這篇文章主要介紹了SQL性能優(yōu)化方法及性能測試,文章圍繞主題展開詳細內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下,希望對你的學習有所幫助
    2022-05-05
  • mysql數(shù)據(jù)庫在表中添加數(shù)據(jù)三種操作方式

    mysql數(shù)據(jù)庫在表中添加數(shù)據(jù)三種操作方式

    這篇文章主要介紹了mysql數(shù)據(jù)庫在表中添加數(shù)據(jù)三種方式,首先創(chuàng)建數(shù)據(jù)庫和表,創(chuàng)建完成后就可以進行添加數(shù)據(jù)的操作了,本文結(jié)合實例代碼給大家介紹的非常詳細,需要的朋友可以參考下
    2023-08-08
  • 通過sql語句將blob里的char取出來轉(zhuǎn)成數(shù)字保存在其它字段

    通過sql語句將blob里的char取出來轉(zhuǎn)成數(shù)字保存在其它字段

    現(xiàn)在需要將blob里地17、18、19三個字段里的數(shù)據(jù)作為數(shù)字保存在blob外新增的三個字段Gem1 Gem2 Gem3上。
    2011-09-09
  • MySQL中CURRENT_TIMESTAMP的使用方式

    MySQL中CURRENT_TIMESTAMP的使用方式

    這篇文章主要介紹了MySQL中CURRENT_TIMESTAMP的使用方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • MySQL Redo與Undo日志詳細解析

    MySQL Redo與Undo日志詳細解析

    這篇文章主要介紹了MySQL Redo與Undo日志詳細解析,Redo日志是物理日志,記錄的是頁面的變化,文章圍繞主題展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的朋友可以參考一下
    2022-07-07
  • MySQL中ONLY_FULL_GROUP_BY模式的使用

    MySQL中ONLY_FULL_GROUP_BY模式的使用

    ONLY_FULL_GROUP_BY是MySQL中一個重要的SQL模式,確保在使用GROUP BY時,所有非聚合函數(shù)列必須在GROUP BY子句中出現(xiàn),避免數(shù)據(jù)歧義和不確定性,下面就來介紹一下具體使用
    2024-09-09
  • 最新版MySQL5.7.19解壓版安裝指南

    最新版MySQL5.7.19解壓版安裝指南

    這篇文章主要介紹了最新版MySQL5.7.19解壓版安裝指南,需要的朋友可以參考下
    2017-08-08
  • 詳解mysql的備份與恢復(fù)

    詳解mysql的備份與恢復(fù)

    這篇文章主要介紹了mysql的備份與恢復(fù)的相關(guān)資料,幫助大家更好的理解和學習mysql,感興趣的朋友可以了解下
    2020-08-08
  • Idea 如何導(dǎo)入Mysql8.0驅(qū)動jar包

    Idea 如何導(dǎo)入Mysql8.0驅(qū)動jar包

    IDEA中的庫(Libraries)就是用來存放外部jar包,我們的項目或模塊需要某些jar包時,可以從這里把包導(dǎo)入到模塊依賴(Dependencies)中,本文給大家介紹Idea 如何導(dǎo)入Mysql8.0驅(qū)動jar包,感興趣的朋友一起看看吧
    2023-12-12
  • mysql索引學習教程

    mysql索引學習教程

    在mysql 中,索引可以分為兩種類型 hash索引和 btree索引。這篇文章主要介紹了mysql索引的相關(guān)知識,非常不錯,具有參考借鑒價值,感興趣的朋友一起看看吧
    2016-09-09

最新評論