首頁技術文章正文

MySQL的性能調(diào)優(yōu)一

更新時間:2018-07-25 來源:黑馬程序員 瀏覽量:

什么是MySQL,怎么安裝,怎么使用,我這里不做說明了。

一、MySQL 與其他數(shù)據(jù)庫的簡單比較


1.1性能比較


        性能方面,一直是MySQL 引以為自豪的一個特點。在權威的第三方評測機構(gòu)多次測試較量各種數(shù)據(jù)庫TPCC 值的過程中,MySQL 一直都有非常優(yōu)異的表現(xiàn),而且在其他所有商用的通用數(shù)據(jù)庫管理系統(tǒng)中,僅僅只有Oracle 數(shù)據(jù)庫能夠與其一較高下。至于各種數(shù)據(jù)庫詳細的性能數(shù)據(jù),我這里就不便記錄,大家完全可以通過網(wǎng)上第三方評測機構(gòu)公布的數(shù)據(jù)了解具體細節(jié)信息。

        MySQL 一直以來奉行一個原則,那就是在保證足夠的穩(wěn)定性的前提下,盡可能的提高自身的處理能力。也就是說,在性能和功能方面,MySQL 第一考慮的要素主要還是性能,MySQL希望自己是一個在滿足客戶99%的功能需求的前提下,花掉剩下的大部分精力來性能努力,而不是希望自己是成為一個比其他任何數(shù)據(jù)庫的功能都要強大的數(shù)據(jù)庫產(chǎn)品。


1.2可靠性


        關于可靠性的比較,并沒有太多詳細的評測比較數(shù)據(jù),但是從目前業(yè)界的交流中可以了解到,幾大商業(yè)廠商的數(shù)據(jù)庫的可靠性肯定是沒有太多值得懷疑的。但是做為開源數(shù)據(jù)庫管理系統(tǒng)的代表,MySQL 也有非常優(yōu)異的表現(xiàn),而并不是像有些人心中所懷疑的那樣,因為不是商業(yè)廠商所提供,就會不夠穩(wěn)定不夠健壯。從當前最火的Facebook 這樣大型的網(wǎng)站都是使用MySQL 數(shù)據(jù)庫,就可以看出,MySQL 在穩(wěn)定可靠性方面,并不會比我們的商業(yè)廠商的產(chǎn)品有太多遜色。而且排在全球前10 位的大型網(wǎng)站里面,大部分都有部分業(yè)務是運行在MySQL數(shù)據(jù)庫環(huán)境上,如Yahoo,Google 等。

        總的來說,MySQL 數(shù)據(jù)庫在發(fā)展過程中一直有自己的三個原則:簡單、高效、可靠。從上面的簡單比較中,我們也可以看出,在MySQL 自己的所有三個原則上面,沒有哪一項是做得不好的。而且,雖然功能并不是MySQL 自身所追求的三個原則之一,但是考慮到當前用戶量的急劇增長,用戶需求越來越越多樣化,MySQL 也不得不在功能方面做出大量的努力,來不斷滿足客戶的新需求。比如最近版本中出現(xiàn)的Eent Scheduler(類似于Oracle 的Job 功能),Partition 功能,自主研發(fā)的Maria 存儲引擎在功能方面的擴展,F(xiàn)alcon 存儲引擎對事務的支持等等,都證明了MySQL 在功能方面也開始了不懈的努力。

        任何一種產(chǎn)品,都不可能是完美的,也不可能適用于所有用戶。我們只有衡量了每一種產(chǎn)品的各種特性之后,從中選擇出一種最適合于自身的產(chǎn)品。

二、MySQL 的主要適用場景


        據(jù)說目前MySQL 用戶已經(jīng)達千萬級別了,其中不乏企業(yè)級用戶??梢哉f是目前最為流行的開源數(shù)據(jù)庫管理系統(tǒng)軟件了。任何產(chǎn)品都不可能是萬能的,也不可能適用于所有的應用場景。


        那么MySQL 到底在什么場景下適用什么場景下不適用呢?


1、Web 網(wǎng)站系統(tǒng)

        Web 站點,是MySQL 最大的客戶群,也是MySQL 發(fā)展史上最為重要的支撐力量,這一點在最開始的MySQL Server 簡介部分就已經(jīng)說明過。

        MySQL 之所以能成為Web 站點開發(fā)者們最青睞的數(shù)據(jù)庫管理系統(tǒng),是因為MySQL 數(shù)據(jù)庫的安裝配置都非常簡單,使用過程中的維護也不像很多大型商業(yè)數(shù)據(jù)庫管理系統(tǒng)那么復雜,而且性能出色。還有一個非常重要的原因就是MySQL 是開放源代碼的,完全可以免費使用。


2、日志記錄系統(tǒng)


        MySQL 數(shù)據(jù)庫的插入和查詢性能都非常的高效,如果設計地較好,在使用MyISAM 存儲引擎的時候,兩者可以做到互不鎖定,達到很高的并發(fā)性能。所以,對需要大量的插入和查詢?nèi)罩居涗浀南到y(tǒng)來說,MySQL 是非常不錯的選擇。比如處理用戶的登錄日志,操作日志等,都是非常適合的應用場景。



3、數(shù)據(jù)倉庫系統(tǒng)

        隨著現(xiàn)在數(shù)據(jù)倉庫數(shù)據(jù)量的飛速增長,我們需要的存儲空間越來越大。數(shù)據(jù)量的不斷增長,使數(shù)據(jù)的統(tǒng)計分析變得越來越低效,也越來越困難。怎么辦?這里有幾個主要的解決思路,一個是采用昂貴的高性能主機以提高計算性能,用高端存儲設備提高I/O 性能,效果理想,但是成本非常高;第二個就是通過將數(shù)據(jù)復制到多臺使用大容量硬盤的廉價pc server上,以提高整體計算性能和I/O 能力,效果尚可,存儲空間有一定限制,成本低廉;第三個,通過將數(shù)據(jù)水平拆分,使用多臺廉價的pc server 和本地磁盤來存放數(shù)據(jù),每臺機器上面都只有所有數(shù)據(jù)的一部分,解決了數(shù)據(jù)量的問題,所有pc server 一起并行計算,也解決了計算能力問題,通過中間代理程序調(diào)配各臺機器的運算任務,既可以解決計算性能問題又可以解決I/O 性能問題,成本也很低廉。在上面的三個方案中,第二和第三個的實現(xiàn),MySQL 都有較大的優(yōu)勢。通過MySQL 的簡單復制功能,可以很好的將數(shù)據(jù)從一臺主機復制到另外一臺,不僅僅在局域網(wǎng)內(nèi)可以復制,在廣域網(wǎng)同樣可以。當然,很多人可能會說,其他的數(shù)據(jù)庫同樣也可以做到,不是只有MySQL 有這樣的功能。確實,很多數(shù)據(jù)庫同樣能做到,但是MySQL是免費的,其他數(shù)據(jù)庫大多都是按照主機數(shù)量或者cpu 數(shù)量來收費,當我們使用大量的pcserver 的時候,license 費用相當驚人。第一個方案,基本上所有數(shù)據(jù)庫系統(tǒng)都能夠?qū)崿F(xiàn),但是其高昂的成本并不是每一個公司都能夠承擔的。


4、嵌入式系統(tǒng)


        嵌入式環(huán)境對軟件系統(tǒng)最大的限制是硬件資源非常有限,在嵌入式環(huán)境下運行的軟件系統(tǒng),必須是輕量級低消耗的軟件。MySQL 在資源的使用方面的伸縮性非常大,可以在資源非常充裕的環(huán)境下運行,也可以在資源非常少的環(huán)境下正常運行。它對于嵌入式環(huán)境來說,是一種非常合適的數(shù)據(jù)庫系統(tǒng),而且MySQL 有專門針對于嵌入式環(huán)境的版本。


三、Query 語句對系統(tǒng)性能的影響


        我想對于各位來說,肯定都清楚SQL 語句的優(yōu)劣是對性能有影響的,但是到底有多大影響可能每個人都會有不同的體會,每個SQL 語句在優(yōu)化之前和優(yōu)化之后的性能差異也是各不相同,所以對于性能差異到底有多大這個問題我們我們這里就不做詳細分析了。我們重點分析實現(xiàn)同樣功能的不同SQL 語句在性能方面會產(chǎn)生較大的差異的根本原因,并通過一個較為典型的示例來對我們的分析做出相應的驗證。

        為什么返回完全相同結(jié)果集的不同SQL 語句,在執(zhí)行性能方面存在差異呢?這里我們先從SQL 語句在數(shù)據(jù)庫中執(zhí)行并獲取所需數(shù)據(jù)這個過程來做一個大概的分析了。

當MySQL Server 的連接線程接收到Client 端發(fā)送過來的SQL 請求之后,會經(jīng)過一系列的分解Parse,進行相應的分析。然后,MySQL 會通過查詢優(yōu)化器模塊(Optimizer)根據(jù)該SQL 所設涉及到的數(shù)據(jù)表的相關統(tǒng)計信息進行計算分析,然后再得出一個MySQL 認為最合理最優(yōu)化的數(shù)據(jù)訪問方式,也就是我們常說的“執(zhí)行計劃”,然后再根據(jù)所得到的執(zhí)行計劃通過調(diào)用存儲引擎借口來獲取相應數(shù)據(jù)。然后再將存儲引擎返回的數(shù)據(jù)進行相關處理,并以Client 端所要求的格式作為結(jié)果集返回給Client 端的應用程序。


        注:這里所說的統(tǒng)計數(shù)據(jù),是我們通過ANALYZE TABLE 命令通知MySQL 對表的相關數(shù)據(jù)做分析之后所獲得到的一些數(shù)據(jù)統(tǒng)計量。這些統(tǒng)計數(shù)據(jù)對MySQL 優(yōu)化器而言是非常重要的,優(yōu)化器所生成的執(zhí)行計劃的好壞,主要就是由這些統(tǒng)計數(shù)據(jù)所決定的。實際上,在其他一些數(shù)據(jù)庫管理軟件中也有類似相應的統(tǒng)計數(shù)據(jù)。

我們都知道,在數(shù)據(jù)庫管理軟件中,最大的性能瓶頸就是在于磁盤IO,也就是數(shù)據(jù)的存取操作上面。而對于同一份數(shù)據(jù),當我們以不同方式去尋找其中的某一點內(nèi)容的時候,所需要讀取的數(shù)據(jù)量可能會有天壤之別,所消耗的資源也自然是區(qū)別甚大。所以,當我們需要從數(shù)據(jù)庫中查詢某個數(shù)據(jù)的時候,所消耗資源的多少主要就取決于數(shù)據(jù)庫以一個什么樣的數(shù)據(jù)讀取方式來完成我們的查詢請求,也就是取決于SQL 語句的執(zhí)行計劃。


        對于唯一一個SQL 語句來說,經(jīng)過MySQL Parse 之后分解的結(jié)構(gòu)都是固定的,只要統(tǒng)計信息穩(wěn)定,其執(zhí)行計劃基本上都是比較固定的。而不同寫法的SQL 語句,經(jīng)過MySQL Parse 之后分解的結(jié)構(gòu)結(jié)構(gòu)就可能完全不同,即使優(yōu)化器使用完全一樣的統(tǒng)計信息來進行優(yōu)化,最后所得出的執(zhí)行計劃也可能完全不一樣。而執(zhí)行計劃又是決定一個SQL 語句最終的資源消耗量的主要因素。所以,實現(xiàn)功能完全一樣的SQL 語句,在性能上面可能會有差別巨大的性能消耗。當然,如果功能一樣,而且經(jīng)過MySQL 的優(yōu)化器優(yōu)化之后的執(zhí)行計劃也完全一致的不同SQL 語句在資源消耗方面可能就相差很小了。當然這里所指的消耗主要是IO 資源的消耗,并不包括CPU 的消耗。

下面我們將通過一兩個具體的示例來分析寫法不一樣而功能完全相同的兩條SQL 的在性能方面的差異。


示例一

        需求:取出某個group(假設id 為100)下的用戶編號(id),用戶昵稱(nick_name)、用戶性別( sexuality ) 、用戶簽名( sign ) 和用戶生日( birthday ) , 并按照加入組的時間(user_group.gmt_create)來進行倒序排列,取出前20 個。


解決方案一、
1532486024325_21.png

解決方案二、
1532486040982_22.png

我們先來看看執(zhí)行計劃:
1532486080904_23.png1532486082857_24.png1532486085685_25.png


        執(zhí)行計劃對比分析:

        解決方案一中的執(zhí)行計劃顯示MySQL 在對兩個參與Join 的表都利用到了索引,user_group 表利用了user_group_gid_ind 索引( key: user_group_gid_ind ) , user 表利用到了主鍵索引( key:PRIMARY),在參與Join 前MySQL 通過Where 過濾后的結(jié)果集與user 表進行Join,最后通過排序取出Join 后結(jié)果的“l(fā)imit 100,20”條結(jié)果返回。

        解決方案二的SQL 語句利用到了子查詢,所以執(zhí)行計劃會稍微復雜一些,首先可以看到兩個表都和解決方案1 一樣都利用到了索引(所使用的索引也完全一樣),執(zhí)行計劃顯示該子查詢以user_group 為驅(qū)動,也就是先通過user_group 進行過濾并馬上進行這一論的結(jié)果集排序,也就取得了SQL 中的“l(fā)imit 100,20”條結(jié)果,然后與user 表進行Join,得到相應的數(shù)據(jù)。這里可能有人會懷疑在自查詢中從user_group表所取得與user 表參與Join的記錄條數(shù)并不是20 條,而是整個group_id=1 的所有結(jié)果。那么清大家看看該執(zhí)行計劃中的第一行,該行內(nèi)容就充分說明了在外層查詢中的所有的20 條記錄全部被返回。 通過比較兩個解決方案的執(zhí)行計劃,我們可以看到第一中解決方案中需要和user 表參與Join 的記錄數(shù)MySQL 通過統(tǒng)計數(shù)據(jù)估算出來是31156,也就是通過user_group 表返回的所有滿足group_id=1 的記錄數(shù)(系統(tǒng)中的實際數(shù)據(jù)是20000)。而第二種解決方案的執(zhí)行計劃中,user 表參與Join 的數(shù)據(jù)就只有20條,兩者相差很大,通過本節(jié)最初的分析,我們認為第二中解決方案應該明顯優(yōu)于第一種解決方案。

        下面我們通過對比兩個解決覺方案的SQL 實際執(zhí)行的profile 詳細信息,來驗證我們上面的判斷。由于SQL 語句執(zhí)行所消耗的最大兩部分資源就是IO和CPU,所以這里為了節(jié)約篇幅,僅列出BLOCK IO 和CPU兩項profile 信息:先打開profiling 功能,然后分別執(zhí)行兩個解決方案的SQL 語句:
1532486105279_26.png1532486109076_27.png1532486112607_28.png


查看系統(tǒng)中的profile 信息,剛剛執(zhí)行的兩個SQL 語句的執(zhí)行profile 信息已經(jīng)記錄下來了:

1532486131405_29.png


1532486169764_30.png1532486172624_31.png        我們先看看兩條SQL 執(zhí)行中的IO 消耗,兩者區(qū)別就在于“Sorting result”,我們回顧一下前面執(zhí)行計劃的對比,兩個解決方案的排序過濾數(shù)據(jù)的時機不一樣,排序后需要取得的數(shù)據(jù)量一個是20000,一個是20,正好和這里的profile 信息吻合,第一種解決方案的“Sorting result”的IO 值是第二種解決方案的將近500 倍。然后再來看看CPU 消耗,所有消耗中,消耗最大的也是“Sorting result”這一項,第一個消耗多出的緣由和上面IO 消耗差異是一樣的。結(jié)論:通過上面兩條功能完全相同的SQL 語句的執(zhí)行計劃分析,以及通過實際執(zhí)行后的profile 數(shù)據(jù)的驗證,都證明了第二種解決方案優(yōu)于第一種解決方案。同時通過后者的實際驗證,也再次證明了我們前面所做的執(zhí)行計劃基本決定了SQL 語句性能。
 

首發(fā):黑馬程序員javaEE培訓學院
首發(fā):http://java.itheima.com/

分享到:
在線咨詢 我要報名
和我們在線交談!