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

MySQL優(yōu)化教程之超大分頁(yè)查詢

 更新時(shí)間:2020年11月11日 11:51:02   作者:god-jiang  
這篇文章主要給大家介紹了關(guān)于MySQL優(yōu)化教程之超大分頁(yè)查詢的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

背景

基本上只要是做后臺(tái)開(kāi)發(fā),都會(huì)接觸到分頁(yè)這個(gè)需求或者功能吧?;旧洗蠹叶际菚?huì)用MySQL的LIMIT來(lái)處理,而且我現(xiàn)在負(fù)責(zé)的項(xiàng)目也是這樣寫(xiě)的。但是一旦數(shù)據(jù)量起來(lái)了,其實(shí)LIMIT的效率會(huì)極其的低,這一篇文章就來(lái)講一下LIMIT子句優(yōu)化的。

LIMIT優(yōu)化

很多業(yè)務(wù)場(chǎng)景都需要用到分頁(yè)這個(gè)功能,基本上都是用LIMIT來(lái)實(shí)現(xiàn)。

建表并且插入200萬(wàn)條數(shù)據(jù):

# 新建一張t5表
CREATE TABLE `t5` (
 `id` int NOT NULL AUTO_INCREMENT,
 `name` varchar(50) NOT NULL,
 `text` varchar(100) NOT NULL,
 PRIMARY KEY (`id`),
 KEY `ix_name` (`name`),
 KEY `ix_test` (`text`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;

# 創(chuàng)建存儲(chǔ)過(guò)程插入200萬(wàn)數(shù)據(jù)
CREATE PROCEDURE t5_insert_200w()
BEGIN
 DECLARE i INT;
 SET i=1000000;
 WHILE i<=3000000 DO
 INSERT INTO t5(`name`,text) VALUES('god-jiang666',concat('text', i));
 SET i=i+1;
 END WHILE;
END;

# 調(diào)用存儲(chǔ)過(guò)程插入200萬(wàn)數(shù)據(jù)
call t5_insert_200w();

在翻頁(yè)比較少的情況下,LIMIT是不會(huì)出現(xiàn)任何性能上的問(wèn)題的。

但是如果用戶需要查到最后面的頁(yè)數(shù)呢?

通常情況下,我們要保證所有的頁(yè)面可以正常跳轉(zhuǎn),因?yàn)椴粫?huì)使用order by xxx desc這樣的倒序SQL來(lái)查詢后面的頁(yè)數(shù),而是采用正序順序來(lái)做分頁(yè)查詢:

select * from t5 order by text limit 100000, 10;

采用這種SQL查詢分頁(yè)的話,從200萬(wàn)數(shù)據(jù)中取出這10行數(shù)據(jù)的代價(jià)是非常大的,需要先排序查出前1000010條記錄,然后拋棄前面1000000條。我的macbook pro跑出來(lái)花了5.578秒。

接下來(lái)我們來(lái)看一下,上面這條SQL語(yǔ)句的執(zhí)行計(jì)劃:

explain select * from t5 order by text limit 1000000, 10;

從執(zhí)行計(jì)劃可以看出,在大分頁(yè)的情況下,MySQL沒(méi)有走索引掃描,即使text字段我已經(jīng)加上了索引。

這是為什么呢?

回到MySQL索引(二)如何設(shè)計(jì)索引中有提及到,MySQL數(shù)據(jù)庫(kù)的查詢優(yōu)化器是采用了基于代價(jià)的,而查詢代價(jià)的估算是基于CPU代價(jià)IO代價(jià)。

如果MySQL在查詢代價(jià)估算中,認(rèn)為全表掃描方式比走索引掃描的方式效率更高的話,就會(huì)放棄索引,直接全表掃描。

這就是為什么在大分頁(yè)的SQL查詢中,明明給該字段加了索引,但是MySQL卻走了全表掃描的原因。

然后我們繼續(xù)用上面的查詢SQL來(lái)驗(yàn)證我的猜想:

explain select * from t5 order by text limit 7774, 10;

explain select * from t5 order by text limit 7775, 10;

以上的實(shí)驗(yàn)均在我的mbp上運(yùn)行的,在7774這個(gè)臨界點(diǎn)上,MySQL分別采用了索引掃描和全表掃描的查詢優(yōu)化方式。

所以可以認(rèn)為MySQL會(huì)根據(jù)它自己的代價(jià)查詢優(yōu)化器來(lái)判斷是否使用索引。

由于MySQL的查詢優(yōu)化器的算法核心是我們無(wú)法人工干預(yù)的,所以我們的優(yōu)化思路就要著手于如何讓分頁(yè)維持在最佳的的分頁(yè)臨界點(diǎn)。

優(yōu)化方式

1、使用覆蓋索引

如果一條SQL語(yǔ)句,通過(guò)索引可以直接獲取查詢的結(jié)果,不再需要回表查詢,就稱這個(gè)索引為覆蓋索引。

在MySQL數(shù)據(jù)庫(kù)中使用explain關(guān)鍵字查看執(zhí)行計(jì)劃,如果extra這一列顯示Using index,就表示這條SQL語(yǔ)句使用了覆蓋索引。

讓我們來(lái)對(duì)比一下使用了覆蓋索引,性能會(huì)提升多少吧。

# 沒(méi)有使用覆蓋索引
select * from t5 order by text limit 1000000, 10;

這次查詢花了3.690秒,讓我們看一下使用了覆蓋索引優(yōu)化會(huì)提升多少性能吧。

# 使用了覆蓋索引
select id, `text` from t5 order by text limit 1000000, 10;

從上面的對(duì)比中,超大分頁(yè)查詢中,使用了覆蓋索引之后,花了0.201秒,而沒(méi)有使用覆蓋索引花了3.690秒,提高了18倍多,這在實(shí)際開(kāi)發(fā)中,就是一個(gè)大的性能優(yōu)化了。(該數(shù)據(jù)在我的mbp上運(yùn)行得出)

2、子查詢優(yōu)化

因?yàn)閷?shí)際開(kāi)發(fā)中,用SELECT查詢一兩列操作是非常少的,因此上述的覆蓋索引的適用范圍就比較有限。

所以我們可以通過(guò)把分頁(yè)的SQL語(yǔ)句改寫(xiě)成子查詢的方法獲得性能上的提升。

select * from t5 where id>=(select id from t5 order by text limit 1000000, 1) limit 10;

其實(shí)使用這種方法,提升的效率和上面使用了覆蓋索引基本一致。

但是這種優(yōu)化方法也有局限性:

  • 這種寫(xiě)法,要求主鍵ID必須是連續(xù)的
  • Where子句不允許再添加其他條件

 3、延遲關(guān)聯(lián)

和上述的子查詢做法類似,我們可以使用JOIN,先在索引列上完成分頁(yè)操作,然后再回表獲取所需要的列。

select a.* from t5 a inner join (select id from t5 order by text limit 1000000, 10) b on a.id=b.id;

從實(shí)驗(yàn)中可以得出,在采用JOIN改寫(xiě)后,上面的兩個(gè)局限性都已經(jīng)解除了,而且SQL的執(zhí)行效率也沒(méi)有損失。

4、記錄上次查詢結(jié)束的位置

和上面使用的方法都不同,記錄上次結(jié)束位置優(yōu)化思路是使用某種變量記錄上一次數(shù)據(jù)的位置,下次分頁(yè)時(shí)直接從這個(gè)變量的位置開(kāi)始掃描,從而避免MySQL掃描大量的數(shù)據(jù)再拋棄的操作。

select * from t5 where id>=1000000 limit 10;

根據(jù)以上實(shí)驗(yàn),不難得出,由于使用了主鍵索引做分頁(yè)操作,SQL的性能是最快的。

總結(jié)

  • 介紹了超大分頁(yè)查詢性能過(guò)差的原因,還有分享了幾個(gè)優(yōu)化思路
  • 超大分頁(yè)的優(yōu)化思路就是讓分頁(yè)的SQL盡量在最佳的性能區(qū)間執(zhí)行,不要觸發(fā)全表掃描即可
  • 希望以上的分享,可以讓你們?cè)贛ySQL這條路上少走彎路~~~

參考資料

  • 《MySQL性能優(yōu)化》第六章 查詢優(yōu)化性能
  • 《數(shù)據(jù)庫(kù)查詢優(yōu)化器的藝術(shù)》

到此這篇關(guān)于MySQL優(yōu)化教程之超大分頁(yè)查詢的文章就介紹到這了,更多相關(guān)MySQL超大分頁(yè)查詢內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • MySQL腳本批量自動(dòng)插入數(shù)據(jù)及數(shù)據(jù)可按條件插入實(shí)現(xiàn)

    MySQL腳本批量自動(dòng)插入數(shù)據(jù)及數(shù)據(jù)可按條件插入實(shí)現(xiàn)

    在初始化數(shù)據(jù)庫(kù)或者導(dǎo)入一些數(shù)據(jù)時(shí),常常會(huì)用到批量的操作,本文主要介紹了MySQL腳本批量自動(dòng)插入數(shù)據(jù)及數(shù)據(jù)可按條件插入實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-01-01
  • 在ubuntu中重置mysql服務(wù)器root密碼的方法

    在ubuntu中重置mysql服務(wù)器root密碼的方法

    在ubuntu下安裝了mysql 5 server,結(jié)果不知道什么原因,安裝時(shí)輸入的root帳號(hào)密碼在使用時(shí)無(wú)論如何都不能通過(guò)數(shù)據(jù)庫(kù)服務(wù)器的驗(yàn)證。無(wú)奈只有重置mysql的root帳號(hào)密碼。查了一下,用了以下方法成功的重置了root帳號(hào)密碼
    2012-10-10
  • MySQL 添加注釋(comment)的方法步驟

    MySQL 添加注釋(comment)的方法步驟

    本文主要介紹了MySQL添加注釋(comment)的方法步驟,字段或列的注釋是用屬性comment來(lái)添加,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-05-05
  • mysql的數(shù)據(jù)壓縮性能對(duì)比詳情

    mysql的數(shù)據(jù)壓縮性能對(duì)比詳情

    這篇文章主要介紹了mysql的數(shù)據(jù)壓縮性能對(duì)比,今天對(duì)這兩種方式分別進(jìn)行了測(cè)試,對(duì)比了二者在磁盤占用以及查詢性能方面各自的優(yōu)劣,下面我們大家一起進(jìn)入文章了解詳細(xì)內(nèi)容,需要的朋友也可以參考一下
    2021-11-11
  • MySQL讀取Binlog日志常見(jiàn)的3種錯(cuò)誤

    MySQL讀取Binlog日志常見(jiàn)的3種錯(cuò)誤

    這篇文章主要介紹了MySQL讀取Binlog日志常見(jiàn)的3種錯(cuò)誤,非常不錯(cuò),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下
    2019-04-04
  • MySQL8.0服務(wù)無(wú)法正常啟動(dòng)的解決過(guò)程

    MySQL8.0服務(wù)無(wú)法正常啟動(dòng)的解決過(guò)程

    這篇文章主要介紹了MySQL8.0服務(wù)無(wú)法正常啟動(dòng)的解決過(guò)程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-12-12
  • MySQL數(shù)據(jù)庫(kù)卸載的完整步驟

    MySQL數(shù)據(jù)庫(kù)卸載的完整步驟

    這篇文章主要為大家詳細(xì)介紹了MySQL數(shù)據(jù)庫(kù)卸載的完整步驟,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-05-05
  • MySQL中符號(hào)@的作用

    MySQL中符號(hào)@的作用

    本文主要介紹了MySQL中符號(hào)@的作用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-06-06
  • mysql 根據(jù)時(shí)間范圍查詢數(shù)據(jù)的操作方法

    mysql 根據(jù)時(shí)間范圍查詢數(shù)據(jù)的操作方法

    這篇文章主要介紹了mysql 根據(jù)時(shí)間范圍查詢數(shù)據(jù)的操作方法,下面是一些常見(jiàn)的時(shí)間范圍查詢示例代碼,需要的朋友可以參考下
    2024-01-01
  • Mysql 5.7.18 解壓版下載安裝及啟動(dòng)mysql服務(wù)的圖文詳解

    Mysql 5.7.18 解壓版下載安裝及啟動(dòng)mysql服務(wù)的圖文詳解

    這篇文章主要介紹了Mysql 5.7.18 解壓版下載安裝及啟動(dòng)mysql服務(wù)的圖文詳解,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2017-05-05

最新評(píng)論