0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

如何優(yōu)化MySQL百萬數(shù)據(jù)的深分頁問題

數(shù)據(jù)分析與開發(fā) ? 來源:數(shù)據(jù)分析與開發(fā) ? 作者:數(shù)據(jù)分析與開發(fā) ? 2022-04-06 15:12 ? 次閱讀

前言

我們?nèi)粘W龇猪撔枨髸r(shí),一般會(huì)用limit實(shí)現(xiàn),但是當(dāng)偏移量特別大的時(shí)候,查詢效率就變得低下。本文將分四個(gè)方案,討論如何優(yōu)化MySQL百萬數(shù)據(jù)的深分頁問題,并附上最近優(yōu)化生產(chǎn)慢SQL的實(shí)戰(zhàn)案例。

limit深分頁為什么會(huì)變慢?

先看下表結(jié)構(gòu)哈:

CREATETABLEaccount(
idint(11)NOTNULLAUTO_INCREMENTCOMMENT'主鍵Id',
namevarchar(255)DEFAULTNULLCOMMENT'賬戶名',
balanceint(11)DEFAULTNULLCOMMENT'余額',
create_timedatetimeNOTNULLCOMMENT'創(chuàng)建時(shí)間',
update_timedatetimeNOTNULLONUPDATECURRENT_TIMESTAMPCOMMENT'更新時(shí)間',
PRIMARYKEY(id),
KEYidx_name(name),
KEYidx_update_time(update_time)//索引
)ENGINE=InnoDBAUTO_INCREMENT=1570068DEFAULTCHARSET=utf8ROW_FORMAT=REDUNDANTCOMMENT='賬戶表';

假設(shè)深分頁的執(zhí)行SQL如下:

selectid,name,balancefromaccountwhereupdate_time>'2020-09-19'limit100000,10;

這個(gè)SQL的執(zhí)行時(shí)間如下:

40531d44-ac98-11ec-aa7f-dac502259ad0.png

執(zhí)行完需要0.742秒,深分頁為什么會(huì)變慢呢?如果換成 limit 0,10,只需要0.006秒哦

40691572-ac98-11ec-aa7f-dac502259ad0.png

我們先來看下這個(gè)SQL的執(zhí)行流程:

  1. 通過普通二級(jí)索引樹idx_update_time,過濾update_time條件,找到滿足條件的記錄ID。
  2. 通過ID,回到主鍵索引樹,找到滿足記錄的行,然后取出展示的列(回表
  3. 掃描滿足條件的100010行,然后扔掉前100000行,返回。
407d71e8-ac98-11ec-aa7f-dac502259ad0.pngSQL的執(zhí)行流程

執(zhí)行計(jì)劃如下:40962e2c-ac98-11ec-aa7f-dac502259ad0.png

SQL變慢原因有兩個(gè)

  1. limit語句會(huì)先掃描offset+n行,然后再丟棄掉前offset行,返回后n行數(shù)據(jù)。也就是說limit 100000,10,就會(huì)掃描100010行,而limit 0,10,只掃描10行。
  2. limit 100000,10 掃描更多的行數(shù),也意味著回表更多的次數(shù)。

通過子查詢優(yōu)化

因?yàn)橐陨系腟QL,回表了100010次,實(shí)際上,我們只需要10條數(shù)據(jù),也就是我們只需要10次回表其實(shí)就夠了。因此,我們可以通過減少回表次數(shù)來優(yōu)化。

回顧B+ 樹結(jié)構(gòu)

那么,如何減少回表次數(shù)呢?我們先來復(fù)習(xí)下B+樹索引結(jié)構(gòu)哈~

InnoDB中,索引分主鍵索引(聚簇索引)和二級(jí)索引

  • 主鍵索引,葉子節(jié)點(diǎn)存放的是整行數(shù)據(jù)
  • 二級(jí)索引,葉子節(jié)點(diǎn)存放的是主鍵的值。
40a51554-ac98-11ec-aa7f-dac502259ad0.png

把條件轉(zhuǎn)移到主鍵索引樹

如果我們把查詢條件,轉(zhuǎn)移回到主鍵索引樹,那就可以減少回表次數(shù)啦。轉(zhuǎn)移到主鍵索引樹查詢的話,查詢條件得改為主鍵id了,之前SQL的update_time這些條件咋辦呢?抽到子查詢那里嘛~

子查詢那里怎么抽的呢?因?yàn)槎?jí)索引葉子節(jié)點(diǎn)是有主鍵ID的,所以我們直接根據(jù)update_time來查主鍵ID即可,同時(shí)我們把 limit 100000的條件,也轉(zhuǎn)移到子查詢,完整SQL如下:

selectid,name,balanceFROMaccountwhereid>=(selecta.idfromaccountawherea.update_time>='2020-09-19'limit100000,1)LIMIT10;寫漏了,可以補(bǔ)下時(shí)間條件在外面

查詢效果一樣的,執(zhí)行時(shí)間只需要0.038秒!

40b9fe4c-ac98-11ec-aa7f-dac502259ad0.png

我們來看下執(zhí)行計(jì)劃40d10f74-ac98-11ec-aa7f-dac502259ad0.png

由執(zhí)行計(jì)劃得知,子查詢 table a查詢是用到了idx_update_time索引。首先在索引上拿到了聚集索引的主鍵ID,省去了回表操作,然后第二查詢直接根據(jù)第一個(gè)查詢的 ID往后再去查10個(gè)就可以了!

40e8d01e-ac98-11ec-aa7f-dac502259ad0.png

因此,這個(gè)方案是可以的~

INNER JOIN 延遲關(guān)聯(lián)

延遲關(guān)聯(lián)的優(yōu)化思路,跟子查詢的優(yōu)化思路其實(shí)是一樣的:都是把條件轉(zhuǎn)移到主鍵索引樹,然后減少回表。不同點(diǎn)是,延遲關(guān)聯(lián)使用了inner join代替子查詢。

優(yōu)化后的SQL如下:

SELECTacct1.id,acct1.name,acct1.balanceFROMaccountacct1INNERJOIN(SELECTa.idFROMaccountaWHEREa.update_time>='2020-09-19'ORDERBYa.update_timeLIMIT100000,10)ASacct2onacct1.id=acct2.id;

查詢效果也是杠桿的,只需要0.034秒

40faef56-ac98-11ec-aa7f-dac502259ad0.png

執(zhí)行計(jì)劃如下:

410e2580-ac98-11ec-aa7f-dac502259ad0.png

查詢思路就是,先通過idx_update_time二級(jí)索引樹查詢到滿足條件的主鍵ID,再與原表通過主鍵ID內(nèi)連接,這樣后面直接走了主鍵索引了,同時(shí)也減少了回表。

標(biāo)簽記錄法

limit 深分頁問題的本質(zhì)原因就是:偏移量(offset)越大,mysql就會(huì)掃描越多的行,然后再拋棄掉。這樣就導(dǎo)致查詢性能的下降。

其實(shí)我們可以采用標(biāo)簽記錄法,就是標(biāo)記一下上次查詢到哪一條了,下次再來查的時(shí)候,從該條開始往下掃描。就好像看書一樣,上次看到哪里了,你就折疊一下或者夾個(gè)書簽,下次來看的時(shí)候,直接就翻到啦。

假設(shè)上一次記錄到100000,則SQL可以修改為:

selectid,name,balanceFROMaccountwhereid>100000orderbyidlimit10;

這樣的話,后面無論翻多少頁,性能都會(huì)不錯(cuò)的,因?yàn)槊辛?code style="font-size:14px;padding:2px 4px;margin-right:2px;margin-left:2px;background-color:rgba(27,31,35,.05);font-family:'Operator Mono', Consolas, Monaco, Menlo, monospace;color:rgb(239,112,96);">id索引。但是這種方式有局限性:需要一種類似連續(xù)自增的字段。

使用between...and...

很多時(shí)候,可以將limit查詢轉(zhuǎn)換為已知位置的查詢,這樣MySQL通過范圍掃描between...and,就能獲得到對(duì)應(yīng)的結(jié)果。

如果知道邊界值為100000,100010后,就可以這樣優(yōu)化:

selectid,name,balanceFROMaccountwhereidbetween100000and100010orderbyid;

手把手實(shí)戰(zhàn)案例

我們一起來看一個(gè)實(shí)戰(zhàn)案例哈。假設(shè)現(xiàn)在有表結(jié)構(gòu)如下,并且有200萬數(shù)據(jù).

CREATETABLEaccount(
idvarchar(32)COLLATEutf8_binNOTNULLCOMMENT'主鍵',
account_novarchar(64)COLLATEutf8_binNOTNULLDEFAULT''COMMENT'賬號(hào)'
amountdecimal(20,2)DEFAULTNULLCOMMENT'金額'
typevarchar(10)COLLATEutf8_binDEFAULTNULLCOMMENT'類型A,B'
create_timedatetimeDEFAULTNULLCOMMENT'創(chuàng)建時(shí)間',
update_timedatetimeDEFAULTNULLCOMMENT'更新時(shí)間',
PRIMARYKEY(id),
KEY`idx_account_no`(account_no),
KEY`idx_create_time`(create_time)
)ENGINE=InnoDBDEFAULTCHARSET=utf8COLLATE=utf8_binCOMMENT='賬戶表'

業(yè)務(wù)需求是這樣:獲取最2021年的A類型賬戶數(shù)據(jù),上報(bào)到大數(shù)據(jù)平臺(tái)。

一般思路的實(shí)現(xiàn)方式

很多伙伴接到這么一個(gè)需求,會(huì)直接這么實(shí)現(xiàn)了:

//查詢上報(bào)總數(shù)量
Integertotal=accountDAO.countAccount();

//查詢上報(bào)總數(shù)量對(duì)應(yīng)的SQL
'countAccount'resultType="java.lang.Integer">
seelctcount(1)
fromaccount
wherecreate_time>='2021-01-010000'
andtype='A'


//計(jì)算頁數(shù)
intpageNo=total%pageSize==0?total/pageSize:(total/pageSize+1);

//分頁查詢,上報(bào)
for(inti=0;ilist=accountDAO.listAccountByPage(startRow,pageSize);
startRow=(pageNo-1)*pageSize;
//上報(bào)大數(shù)據(jù)
postBigData(list);
}

//分頁查詢SQL(可能存在limit深分頁問題,因?yàn)閍ccount表數(shù)據(jù)量幾百萬)
'listAccountByPage'>
seelct*
fromaccount
wherecreate_time>='2021-01-010000'
andtype='A'
limit#{startRow},#{pageSize}

實(shí)戰(zhàn)優(yōu)化方案

以上的實(shí)現(xiàn)方案,會(huì)存在limit深分頁問題,因?yàn)閍ccount表數(shù)據(jù)量幾百萬。那怎么優(yōu)化呢?

其實(shí)可以使用標(biāo)簽記錄法,有些伙伴可能會(huì)有疑惑,id主鍵不是連續(xù)的呀,真的可以使用標(biāo)簽記錄?

當(dāng)然可以,id不是連續(xù),我們可以通過order by讓它連續(xù)嘛。優(yōu)化方案如下:

//查詢最小ID
StringlastId=accountDAO.queryMinId();

//查詢最小ID對(duì)應(yīng)的SQL
"queryMinId"returnType=“java.lang.String”>
selectMIN(id)
fromaccount
wherecreate_time>='2021-01-010000'
andtype='A'


//一頁的條數(shù)
IntegerpageSize=100;

Listlist;
do{
list=listAccountByPage(lastId,pageSize);
//標(biāo)簽記錄法,記錄上次查詢過的Id
lastId=list.get(list,size()-1).getId();
//上報(bào)大數(shù)據(jù)
postBigData(list);
}while(CollectionUtils.isNotEmpty(list));

"listAccountByPage">
select*
fromaccount
wherecreate_time>='2021-01-010000'
andid>#{lastId}
andtype='A'
orderbyidasc
limit#{pageSize}


審核編輯 :李倩


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • SQL
    SQL
    +關(guān)注

    關(guān)注

    1

    文章

    750

    瀏覽量

    43900
  • MySQL
    +關(guān)注

    關(guān)注

    1

    文章

    789

    瀏覽量

    26283

原文標(biāo)題:聊聊如何解決 MySQL 深分頁問題

文章出處:【微信號(hào):DBDevs,微信公眾號(hào):數(shù)據(jù)分析與開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    MySQL的執(zhí)行過程 SQL語句性能優(yōu)化常用策略

    回顧 MySQL 的執(zhí)行過程,幫助介紹如何進(jìn)行 sql 優(yōu)化
    的頭像 發(fā)表于 12-12 10:26 ?536次閱讀
    <b class='flag-5'>MySQL</b>的執(zhí)行過程 SQL語句性能<b class='flag-5'>優(yōu)化</b>常用策略

    MySQL執(zhí)行過程:如何進(jìn)行sql 優(yōu)化

    (1)客戶端發(fā)送一條查詢語句到服務(wù)器; (2)服務(wù)器先查詢緩存,如果命中緩存,則立即返回存儲(chǔ)在緩存中的數(shù)據(jù); (3)未命中緩存后,MySQL 通過關(guān)鍵字將 SQL 語句進(jìn)行解析,并生成一顆對(duì)應(yīng)的解析樹,MySQL 解析器將使用
    的頭像 發(fā)表于 12-12 10:19 ?333次閱讀
    <b class='flag-5'>MySQL</b>執(zhí)行過程:如何進(jìn)行sql <b class='flag-5'>優(yōu)化</b>

    eclipse怎么連接數(shù)據(jù)mysql

    連接Eclipse和MySQL數(shù)據(jù)庫可以通過JDBC(Java Database Connectivity)來實(shí)現(xiàn)。以下是詳細(xì)步驟: 下載并安裝MySQL數(shù)據(jù)庫:你首先需要下載并安裝
    的頭像 發(fā)表于 12-06 11:06 ?1026次閱讀

    MySQL數(shù)據(jù)庫的url地址

    MySQL數(shù)據(jù)庫的URL地址是用于連接到MySQL服務(wù)器的地址。URL是一種統(tǒng)一資源定位符,用于指定特定資源的位置和訪問方式。MySQL數(shù)據(jù)
    的頭像 發(fā)表于 12-06 10:58 ?2115次閱讀

    mysql數(shù)據(jù)庫基礎(chǔ)命令

    MySQL是一個(gè)流行的關(guān)系型數(shù)據(jù)庫管理系統(tǒng),經(jīng)常用于存儲(chǔ)、管理和操作數(shù)據(jù)。在本文中,我們將詳細(xì)介紹MySQL的基礎(chǔ)命令,并提供與每個(gè)命令相關(guān)的詳細(xì)解釋。 登錄
    的頭像 發(fā)表于 12-06 10:56 ?436次閱讀

    mybatis邏輯分頁和物理分頁的區(qū)別

    這兩種分頁方式的區(qū)別。 邏輯分頁是在數(shù)據(jù)庫中執(zhí)行查詢時(shí)使用的一種分頁方式。這種方式是通過在查詢語句中添加LIMIT或OFFSET關(guān)鍵字來限制結(jié)果集的大小和偏移量來實(shí)現(xiàn)的。常見的邏輯
    的頭像 發(fā)表于 12-03 14:54 ?726次閱讀

    mysql中的數(shù)據(jù)大于千萬怎么辦

    當(dāng)MySQL中的數(shù)據(jù)量達(dá)到千萬級(jí)別時(shí),為了保證數(shù)據(jù)庫的性能和穩(wěn)定性,需要采取一系列優(yōu)化措施和架構(gòu)設(shè)計(jì)。在本文中,我將詳細(xì)介紹如何應(yīng)對(duì)大規(guī)模數(shù)據(jù)
    的頭像 發(fā)表于 11-23 14:41 ?1283次閱讀

    MySQL性能優(yōu)化方法

    MySQL 性能優(yōu)化是一項(xiàng)關(guān)鍵的任務(wù),可以提高數(shù)據(jù)庫的運(yùn)行速度和效率。以下是一些優(yōu)化方法,包括具體代碼和詳細(xì)優(yōu)化方案。
    的頭像 發(fā)表于 11-22 09:59 ?445次閱讀

    MySQL數(shù)據(jù)庫基礎(chǔ)知識(shí)

    MySQL 是一種開源的關(guān)系型數(shù)據(jù)庫管理系統(tǒng),它是目前最流行的數(shù)據(jù)庫之一。MySQL 提供了一種結(jié)構(gòu)化的方法來管理大量的數(shù)據(jù),并且具有高效、
    的頭像 發(fā)表于 11-21 11:09 ?834次閱讀

    mysql和sql server區(qū)別

    MySQL和SQL Server是兩種常見的關(guān)系型數(shù)據(jù)庫管理系統(tǒng)(RDBMS),用于存儲(chǔ)和管理數(shù)據(jù)庫。雖然它們都支持SQL語言,但在其他方面存在一些顯著的區(qū)別。以下是MySQL和SQL
    的頭像 發(fā)表于 11-21 11:07 ?1246次閱讀

    MySQL導(dǎo)出的步驟

    MySQL是一種常用的關(guān)系型數(shù)據(jù)庫管理系統(tǒng),用于存儲(chǔ)和管理大量的結(jié)構(gòu)化數(shù)據(jù)。在實(shí)際應(yīng)用中,我們經(jīng)常需要將MySQL數(shù)據(jù)庫中的
    的頭像 發(fā)表于 11-21 10:58 ?662次閱讀

    數(shù)據(jù)mysql基本增刪改查

    MySQL是一種開源的關(guān)系型數(shù)據(jù)庫管理系統(tǒng),常用于Web應(yīng)用程序的數(shù)據(jù)存儲(chǔ)和管理。通過使用MySQL,用戶可以進(jìn)行數(shù)據(jù)的增刪改查操作,從而實(shí)
    的頭像 發(fā)表于 11-16 16:35 ?1352次閱讀

    mysql是一個(gè)什么類型的數(shù)據(jù)

    MySQL是一種關(guān)系型數(shù)據(jù)庫管理系統(tǒng)(RDBMS),用于存儲(chǔ)和管理大量結(jié)構(gòu)化數(shù)據(jù)。它被廣泛用于各種應(yīng)用程序和網(wǎng)站的后端,包括電子商務(wù)平臺(tái)、社交媒體網(wǎng)站、金融系統(tǒng)等等。MySQL的特點(diǎn)是
    的頭像 發(fā)表于 11-16 14:43 ?1431次閱讀

    如何提高Mysql數(shù)據(jù)庫的訪問瓶頸

    為了提高Mysql數(shù)據(jù)庫的訪問瓶頸,常用的方法有如下兩個(gè): 在服務(wù)器端增加緩存服務(wù)器緩存常用的數(shù)據(jù)(例如redis) 增加連接池,來提高MYsql Server的訪問效率,在高并發(fā)的
    的頭像 發(fā)表于 11-08 16:22 ?931次閱讀
    如何提高<b class='flag-5'>Mysql</b><b class='flag-5'>數(shù)據(jù)</b>庫的訪問瓶頸

    如何在Rust中連接和使用MySQL數(shù)據(jù)

    MySQL是一個(gè)廣泛使用的關(guān)系型數(shù)據(jù)庫,Rust作為一門相對(duì)較新的系統(tǒng)級(jí)編程語言,具有C語言般的高性能、安全、并發(fā)等特性,因此與MySQL一起使用是一種非常有趣的選擇。在本教程中,我們將手把手地展示
    的頭像 發(fā)表于 09-30 17:05 ?1403次閱讀