日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 單片機(jī) > 架構(gòu)師社區(qū)
[導(dǎo)讀]我們都知道,業(yè)務(wù)開發(fā)涉及到數(shù)據(jù)庫的SQL操作時(shí),一定要review是否命中索引。否則,會(huì)走全表掃描,如果表數(shù)據(jù)量很大時(shí),會(huì)慢的要死。假如命中了索引呢?是不是就不會(huì)有慢查詢?殊不知,我們習(xí)以為常的常識(shí)有時(shí)也會(huì)誤導(dǎo)我們!人生好難!聊這個(gè)話題,要有一定技術(shù)基礎(chǔ),需了解B樹的存儲(chǔ)結(jié)構(gòu)如果...

我們都知道,業(yè)務(wù)開發(fā)涉及到數(shù)據(jù)庫的SQL操作時(shí),一定要 review 是否命中索引。否則,會(huì)走 全表掃描,如果表數(shù)據(jù)量很大時(shí),會(huì)慢的要死。

假如命中了索引呢?是不是就不會(huì)有慢查詢?


同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


殊不知,我們習(xí)以為常的常識(shí)有時(shí)也會(huì)誤導(dǎo)我們!

人生好難!

聊這個(gè)話題,要有一定技術(shù)基礎(chǔ),需了解 B 樹的存儲(chǔ)結(jié)構(gòu)

如果不是很清楚的話,先看下之前一篇文章,有詳細(xì)介紹

面試題:mysql 一棵 B 樹可以存多少條數(shù)據(jù)?


1、工作準(zhǔn)備:建表,造數(shù)據(jù)


首先創(chuàng)建一張 user 表,并創(chuàng)建一個(gè) id的主鍵索引,和一個(gè) user_name 的普通索引。

CREATE?TABLE?`user`?(
??`id`?bigint(20)?NOT?NULL?AUTO_INCREMENT,
??`user_name`?varchar(128)?NOT?NULL?DEFAULT?''?COMMENT?'用戶名',
??`age`?int(11)?NOT?NULL??COMMENT?'年齡',
??`address`?varchar(128)?COMMENT?'地址',
???PRIMARY?KEY?(`id`),
???key?`idx_user_name`?(user_name),
)?ENGINE=InnoDB??DEFAULT?CHARSET=utf8mb4?COMMENT='用戶表';

啟動(dòng)程序,往 user 表中插入 10000 條數(shù)據(jù)。

@GetMapping("/insert_batch")
public?Object?insertBatch(@RequestParam("batch")?int?batch)?{
????for?(int?j?=?1;?j?<=?batch;?j )?{
????????List?userList?=?new?ArrayList<>();
????????for?(int?i?=?1;?i?<=?100;?i )?{
????????????User?user?=?User.builder().userName("Tom哥-"? ?((j?-?1)?*?100? ?i)).age(29).address("上海").build();
????????????userList.add(user);
????????}
????????userMapper.insertBatch(userList);
????}
????return?"success";
}


2、慢查詢


在分析原因前,我們先來了解 mysql 慢查詢是什么?如何定義的?

慢查詢定義:

MySQL的慢查詢?nèi)罩臼荕ySQL提供的一種日志記錄,用來記錄在MySQL中響應(yīng)時(shí)間超過閥值的語句,具體指運(yùn)行時(shí)間超過long_query_time值的SQL,則會(huì)被記錄到慢查詢?nèi)罩局小?/p>
慢查詢相關(guān)參數(shù):

  • slow_query_log:是否開啟慢查詢?nèi)罩荆?表示開啟,0表示關(guān)閉。
  • log-slow-queries:舊版(5.6以下版本)MySQL數(shù)據(jù)庫慢查詢?nèi)罩敬鎯?chǔ)路徑??梢圆辉O(shè)置該參數(shù),系統(tǒng)則會(huì)默認(rèn)給一個(gè)缺省的文件host_name-slow.log
  • slow-query-log-file:新版(5.6及以上版本)MySQL數(shù)據(jù)庫慢查詢?nèi)罩敬鎯?chǔ)路徑??梢圆辉O(shè)置該參數(shù),系統(tǒng)則會(huì)默認(rèn)給一個(gè)缺省的文件host_name-slow.log
  • long_query_time:慢查詢閾值,當(dāng)查詢時(shí)間高于設(shè)定的閾值時(shí),記錄到日志
  • log_queries_not_using_indexes:未使用索引的查詢也被記錄到慢查詢?nèi)罩局校蛇x項(xiàng))
默認(rèn)情況下slow_query_log的值為OFF,表示慢查詢?nèi)罩臼墙玫?,可以通過設(shè)置slow_query_log的值來開啟,如下所示:


同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?

使用set global slow_query_log=1 開啟了慢查詢?nèi)罩局粚Ξ?dāng)前數(shù)據(jù)庫生效,如果MySQL重啟后則會(huì)失效。如果要永久生效,必須修改配置文件 my.cnf

long_query_time的默認(rèn)值為10 秒,支持二次修改。線上我們一般會(huì)設(shè)置成1秒,如果業(yè)務(wù)對延遲敏感的話,我們根據(jù)需要設(shè)置一個(gè)更低的值。


同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


3、開始實(shí)驗(yàn)



首先看下以下幾種場景的SQL語句執(zhí)行時(shí),索引的命中情況。


1、執(zhí)行explain select * from user;,發(fā)現(xiàn) key 這列為NULL,說明了沒有命中索引,走了全表掃描。

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


2、執(zhí)行 explain select * from user where id=10;,發(fā)現(xiàn) key 這列為 PRIMARY,說明使用了主鍵索引。

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


3、執(zhí)行 explain select user_name from user;,發(fā)現(xiàn) key 這列為 idx_user_name,說明使用了二級(jí)普通索引。

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


但是,實(shí)驗(yàn)發(fā)現(xiàn),雖然走了二級(jí)索引,但是 rows 掃描行為 9968,說明走了全表掃描。性能很差。

本文測試只造了 1W 條數(shù)據(jù),如果線上環(huán)境有個(gè)千萬級(jí)數(shù)據(jù)量,那估計(jì)要好幾秒才能響應(yīng)結(jié)果。

如果請求并發(fā)量很高,很容易引發(fā)數(shù)據(jù)庫連接無法及時(shí)釋放,導(dǎo)致客戶端無法獲取數(shù)據(jù)庫連接而報(bào)錯(cuò)。


4、命中索引,依然很慢


我們知道所有的數(shù)據(jù)都是存儲(chǔ)在 B 索引樹上,當(dāng)執(zhí)行 explain select * from user where id>0; 時(shí),發(fā)現(xiàn)使用了主鍵索引。

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?

mysql 優(yōu)化器根據(jù)主鍵索引找到第一個(gè) id>0 的值,雖然走了索引但其實(shí)還是全表掃描。

沒命中索引會(huì)走全表掃描,命中了索引也可能走全表掃描。

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?

看來是否命中索引,并不是評判 SQL 性能好壞的唯一標(biāo)準(zhǔn)。

其實(shí),還有一個(gè)重要指標(biāo),那就是 掃描行數(shù)。

當(dāng)一個(gè)表很大時(shí),不僅要關(guān)注是否有索引,還要關(guān)注索引的過濾性是否足夠好。


5、回表優(yōu)化


首先為user表 增加一個(gè) user_nameage 的聯(lián)合索引。

ALTER?TABLE?`user`?ADD?INDEX?idx_user_name_age?(?`user_name`,`age`?);
同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


執(zhí)行 explain select * from user where user_name like 'Tom哥-1%' and age =29;

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


執(zhí)行流程:

  • ① 首先在 idx_user_name_age 索引樹,查找第一個(gè)以 Tom哥-1 開頭的記錄對應(yīng)的主鍵id
  • ② 根據(jù)主鍵id從主鍵索引樹找到整行記錄,并根據(jù)age做判斷過濾,等于29則留下,否則丟棄。這個(gè)過程也稱為回表
  • ③ 然后,在 idx_user_name_age 聯(lián)合索引樹上向右遍歷,找到下一個(gè)主鍵id
  • ④ 再執(zhí)行第二步
  • ⑤ 后面重復(fù)執(zhí)行第三步、第四步,直到user_name不是以 Tom哥-1 開頭,則結(jié)束
  • ⑥ 返回所有查詢結(jié)果
分析:

由于按user_name 的前綴匹配,idx_user_name_age二級(jí)索引中的 age 部分并沒有發(fā)揮作用。導(dǎo)致了大量回表查詢,性能較差。

有什么優(yōu)化策略:

MySQL 5.6 版本引入一個(gè) Index Condition Pushdown Optimization

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


https://dev.mysql.com/doc/refman/5.6/en/index-condition-pushdown-optimization.html

優(yōu)化后,執(zhí)行流程:

  • ① 首先在 idx_user_name_age 索引樹,查找第一個(gè)以 Tom哥-1 開頭的索引記錄
  • ② 然后,判斷這個(gè)索引記錄中的 age 是否等于 29。如果是,回表 取出整行數(shù)據(jù),作為后面的結(jié)果返回;如果不是,則丟棄
  • ③ 在 idx_user_name_age 聯(lián)合索引樹上向右遍歷,重復(fù)第二步,直到user_name不是以 Tom哥-1 開頭,則結(jié)束
  • ④ 返回所有查詢結(jié)果
跟上面的過程差別,在于判斷 age 是否等于 29 放在了遍歷聯(lián)合索引過程中進(jìn)行,不需要回表判斷,大大降低了回表的次數(shù),提升性能。

當(dāng)然這個(gè)優(yōu)化依然沒有繞開最左前綴原則,索引的過濾性仍然有提升空間。

這時(shí),我們需要引入一個(gè)叫 虛擬列 的概念。

修改表結(jié)構(gòu):

ALTER?TABLE?`user`?add?user_name_first?varchar(12)?generated?always?as?
(left(user_name,6))?,?add?index(user_name_first,age);
同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


執(zhí)行 explain select * from user where user_name_first like 'Tom哥-1%' and age =29;

同事問我,SQL?語句明明命中了索引,為什么執(zhí)行很慢?


比較發(fā)現(xiàn),掃描行數(shù) row 變小了,證明優(yōu)化有效果。


6、寫在最后

slow_query_log 收集到的慢 SQL ,結(jié)合 explain 分析是否命中索引,結(jié)合掃描行數(shù),有針對性的優(yōu)化慢 SQL。

但是要注意一點(diǎn),慢 SQL 日志中也可能有正常的 SQL,可能只是當(dāng)時(shí)CPU等系統(tǒng)資源過載,影響到正常 SQL 的執(zhí)行速度。

簡單來講,慢查詢和索引沒有必然聯(lián)系,一個(gè)SQL語句的執(zhí)行效率最終要看的是掃描行數(shù)。另外可以使用虛擬列和聯(lián)合索引來提升復(fù)雜查詢的執(zhí)行效率。


本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動(dòng)電源

在工業(yè)自動(dòng)化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動(dòng)力設(shè)備,其驅(qū)動(dòng)電源的性能直接關(guān)系到整個(gè)系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動(dòng)勢抑制與過流保護(hù)是驅(qū)動(dòng)電源設(shè)計(jì)中至關(guān)重要的兩個(gè)環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動(dòng)性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動(dòng)電源

LED 驅(qū)動(dòng)電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個(gè)照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動(dòng)電源易損壞的問題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動(dòng)電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動(dòng)電源的公式,電感內(nèi)電流波動(dòng)大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動(dòng)電源

電動(dòng)汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動(dòng)汽車的核心技術(shù)之一是電機(jī)驅(qū)動(dòng)控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動(dòng)系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動(dòng)汽車的動(dòng)力性能和...

關(guān)鍵字: 電動(dòng)汽車 新能源 驅(qū)動(dòng)電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動(dòng)電源 LED

LED通用照明設(shè)計(jì)工程師會(huì)遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動(dòng)電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動(dòng)電源的電磁干擾(EMI)問題成為了一個(gè)不可忽視的挑戰(zhàn)。電磁干擾不僅會(huì)影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動(dòng)電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動(dòng)電源

關(guān)鍵字: LED 驅(qū)動(dòng)電源 開關(guān)電源

LED驅(qū)動(dòng)電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動(dòng)LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動(dòng)電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動(dòng)電源
關(guān)閉