作者:soゝso
原文地址:https://www.sojson.com/blog/41.html
MySQL 數(shù)據(jù)庫(kù)性能的優(yōu)化是 MySQL 數(shù)據(jù)庫(kù)發(fā)展的必經(jīng)之路, MySQL 數(shù)據(jù)庫(kù)性能的優(yōu)化也是 MySQL 數(shù)據(jù)庫(kù)前進(jìn)的見(jiàn)證,下文中將從從4個(gè)方面給出了 MySQL 數(shù)據(jù)庫(kù)性能優(yōu)化的方法。
MySQL 可以很好的支持大數(shù)據(jù)量的存取,但是一般說(shuō)來(lái),數(shù)據(jù)庫(kù)中的表越小,在它上面執(zhí)行的查詢也就會(huì)越快。因此,在創(chuàng)建表的時(shí)候,為了獲得更好的性能,我們可以將表中字段的寬度設(shè)得盡可能小。例如,在定義郵政編碼這個(gè)字段時(shí),如果將其設(shè)置為CHAR(255),
顯然給數(shù)據(jù)庫(kù)增加了不必要的空間,甚至使用VARCHAR
這種類型也是多余的,因?yàn)?code>CHAR(6)就可以很好的完成任務(wù)了。同樣的,如果可以的話,我們應(yīng)該使用MEDIUMINT
而不是BIGIN
來(lái)定義整型字段。
另外一個(gè)提高效率的方法是在可能的情況下,應(yīng)該盡量把字段設(shè)置為NOT NULL
,這樣在將來(lái)執(zhí)行查詢的時(shí)候,數(shù)據(jù)庫(kù)不用去比較NULL
值。
對(duì)于某些文本字段,例如“省份”或者“性別”,我們可以將它們定義為ENUM
類型。因?yàn)樵?nbsp;MySQL 中,ENUM
類型被當(dāng)作數(shù)值型數(shù)據(jù)來(lái)處理,而數(shù)值型數(shù)據(jù)被處理起來(lái)的速度要比文本類型快得多。這樣,我們又可以提高數(shù)據(jù)庫(kù)的性能。
MySQL 從4.1開(kāi)始支持 SQL 的子查詢。這個(gè)技術(shù)可以使用SELECT
語(yǔ)句來(lái)創(chuàng)建一個(gè)單列的查詢結(jié)果,然后把這個(gè)結(jié)果作為過(guò)濾條件用在另一個(gè)查詢中。例如,我們要將客戶基本信息表中沒(méi)有任何訂單的客戶刪除掉,就可以利用子查詢先從銷售信息表中將所有發(fā)出訂單的客戶ID
取出來(lái),然后將結(jié)果傳遞給主查詢,如下所示:
DELETE FROM customerinfo
WHERE CustomerID NOT in (SELECT CustomerID FROM salesinfo );
使用子查詢可以一次性的完成很多邏輯上需要多個(gè)步驟才能完成的 SQL 操作,同時(shí)也可以避免事務(wù)或者表鎖死,并且寫(xiě)起來(lái)也很容易。但是,有些情況下,子查詢可以被更有效率的連接(JOIN
).. 替代。例如,假設(shè)我們要將所有沒(méi)有訂單記錄的用戶取出來(lái),可以用下面這個(gè)查詢完成:
SELECT * FROM customerinfo
WHERE CustomerID NOT in (SELECT CustomerID FROM salesinfo )
如果使用連接(JOIN
).. 來(lái)完成這個(gè)查詢工作,速度將會(huì)快很多。尤其是當(dāng)salesinfo
表中對(duì)CustomerID
建有索引的話,性能將會(huì)更好,查詢?nèi)缦拢?/p>
SELECT * FROM customerinfo
LEFT JOIN salesinfo ON customerinfo.CustomerID=salesinfo.CustomerID
WHERE salesinfo.CustomerID IS NULL
連接(JOIN
).. 之所以更有效率一些,是因?yàn)?nbsp; MySQL 不需要在內(nèi)存中創(chuàng)建臨時(shí)表來(lái)完成這個(gè)邏輯上的需要兩個(gè)步驟的查詢工作。
UNION
)來(lái)代替手動(dòng)創(chuàng)建的臨時(shí)表MySQL 從 4.0 的版本開(kāi)始支持 UNION
查詢,它可以把需要使用臨時(shí)表的兩條或更多的 SELECT
查詢合并的一個(gè)查詢中。在客戶端的查詢會(huì)話結(jié)束的時(shí)候,臨時(shí)表會(huì)被自動(dòng)刪除,從而保證數(shù)據(jù)庫(kù)整齊、高效。使用 UNION
來(lái)創(chuàng)建查詢的時(shí)候,我們只需要用 UNION
作為關(guān)鍵字把多個(gè) SELECT
語(yǔ)句連接起來(lái)就可以了,要注意的是所有 SELECT
語(yǔ)句中的字段數(shù)目要想同。下面的例子就演示了一個(gè)使用 UNION
的查詢。
SELECT Name, Phone FROM client
UNION
SELECT Name, BirthDate FROM author
UNION
SELECT Name, Supplier FROM product
盡管我們可以使用子查詢(Sub-Queries
)、連接(JOIN
)和聯(lián)合(UNION
)來(lái)創(chuàng)建各種各樣的查詢,但不是所有的數(shù)據(jù)庫(kù)操作都可以只用一條或少數(shù)幾條 SQL 語(yǔ)句就可以完成的。更多的時(shí)候是需要用到一系列的語(yǔ)句來(lái)完成某種工作。但是在這種情況下,當(dāng)這個(gè)語(yǔ)句塊中的某一條語(yǔ)句運(yùn)行出錯(cuò)的時(shí)候,整個(gè)語(yǔ)句塊的操作就會(huì)變得不確定起來(lái)。設(shè)想一下,要把某個(gè)數(shù)據(jù)同時(shí)插入兩個(gè)相關(guān)聯(lián)的表中,可能會(huì)出現(xiàn)這樣的情況:第一個(gè)表中成功更新后,數(shù)據(jù)庫(kù)突然出現(xiàn)意外狀況,造成第二個(gè)表中的操作沒(méi)有完成,這樣,就會(huì)造成數(shù)據(jù)的不完整,甚至?xí)茐臄?shù)據(jù)庫(kù)中的數(shù)據(jù)。要避免這種情況,就應(yīng)該使用事務(wù),它的作用是:要么語(yǔ)句塊中每條語(yǔ)句都操作成功,要么都失敗。換句話說(shuō),就是可以保持?jǐn)?shù)據(jù)庫(kù)中數(shù)據(jù)的一致性和完整性。事物以BEGIN
關(guān)鍵字開(kāi)始,COMMIT
關(guān)鍵字結(jié)束。在這之間的一條 SQL 操作失敗,那么,ROLLBACK
命令就可以把數(shù)據(jù)庫(kù)恢復(fù)到BEGIN
開(kāi)始之前的狀態(tài)。
BEGIN;
INSERT INTO salesinfo SET CustomerID=14;
UPDATE inventory SET Quantity=11
WHERE item='book';
COMMIT;
事務(wù)的另一個(gè)重要作用是當(dāng)多個(gè)用戶同時(shí)使用相同的數(shù)據(jù)源時(shí),它可以利用鎖定數(shù)據(jù)庫(kù)的方法來(lái)為用戶提供一種安全的訪問(wèn)方式,這樣可以保證用戶的操作不被其它的用戶所干擾。
通過(guò)上文中介紹的方法就能夠?qū)崿F(xiàn) MySQL 數(shù)據(jù)庫(kù)的性能的優(yōu)化工作,希望大家都能夠從上文中有所收獲。
聯(lián)系客服