MySQL性能優(yōu)化的最佳20+條經(jīng)驗

MySQL性能優(yōu)化的最佳20+條經(jīng)驗

今天,數(shù)據(jù)庫的操作越來越成為整個應(yīng)用的性能瓶頸了,這點(diǎn)對于Web應(yīng)用尤其明顯哼凯。關(guān)于數(shù)據(jù)庫的性能,這并不只是DBA才需要擔(dān)心的事楚里,而這更是我 們程序員需要去關(guān)注的事情断部。當(dāng)我們?nèi)ピO(shè)計數(shù)據(jù)庫表結(jié)構(gòu),對操作數(shù)據(jù)庫時(尤其是查表時的SQL語句)腻豌,我們都需要注意數(shù)據(jù)操作的性能家坎。這里嘱能,我們不會講過 多的SQL語句的優(yōu)化,而只是針對MySQL這一Web應(yīng)用最多的數(shù)據(jù)庫虱疏。希望下面的這些優(yōu)化技巧對你有用惹骂。

1. 為查詢緩存優(yōu)化你的查詢

大多數(shù)的MySQL服務(wù)器都開啟了查詢緩存。這是提高性最有效的方法之一做瞪,而且這是被MySQL的數(shù)據(jù)庫引擎處理的对粪。當(dāng)有很多相同的查詢被執(zhí)行了多次的時候,這些查詢結(jié)果會被放到一個緩存中装蓬,這樣著拭,后續(xù)的相同的查詢就不用操作表而直接訪問緩存結(jié)果了。

這里最主要的問題是牍帚,對于程序員來說儡遮,這個事情是很容易被忽略的。因為暗赶,我們某些查詢語句會讓MySQL不使用緩存鄙币。請看下面的示例://?查詢緩存不開啟

$r=mysql_query("SELECT?username?FROM?user?WHERE?signup_date?>=?CURDATE()");

//?開啟查詢緩存

$today=date("Y-m-d");

$r=mysql_query("SELECT?username?FROM?user?WHERE?signup_date?>=?'$today'");

復(fù)制代碼上面兩條SQL語句的差別就是 CURDATE() ,MySQL的查詢緩存對這個函數(shù)不起作用蹂随。所以十嘿,像 NOW() 和 RAND() 或是其它的諸如此類的SQL函數(shù)都不會開啟查詢緩存,因為這些函數(shù)的返回是會不定的易變的岳锁。所以绩衷,你所需要的就是用一個變量來代替MySQL的函數(shù),從而 開啟緩存激率。

2. EXPLAIN 你的 SELECT 查詢

使用 EXPLAIN 關(guān)鍵字可以讓你知道MySQL是如何處理你的SQL語句的咳燕。這可以幫你分析你的查詢語句或是表結(jié)構(gòu)的性能瓶頸。

EXPLAIN 的查詢結(jié)果還會告訴你你的索引主鍵被如何利用的柱搜,你的數(shù)據(jù)表是如何被搜索和排序的……等等迟郎,等等剥险。

挑一個你的SELECT語句(推薦挑選那個最復(fù)雜的聪蘸,有多表聯(lián)接的),把關(guān)鍵字EXPLAIN加到前面表制。你可以使用phpmyadmin來做這個事健爬。然后,你會看到一張表格么介。下面的這個示例中娜遵,我們忘記加上了group_id索引,并且有表聯(lián)接:

當(dāng)我們?yōu)?group_id 字段加上索引后:

我們可以看到壤短,前一個結(jié)果顯示搜索了 7883 行设拟,而后一個只是搜索了兩個表的 9 和 16 行慨仿。查看rows列可以讓我們找到潛在的性能問題。

3. 當(dāng)只要一行數(shù)據(jù)時使用 LIMIT 1

當(dāng)你查詢表的有些時候纳胧,你已經(jīng)知道結(jié)果只會有一條結(jié)果镰吆,但因為你可能需要去fetch游標(biāo),或是你也許會去檢查返回的記錄數(shù)跑慕。

在這種情況下万皿,加上 LIMIT 1 可以增加性能。這樣一樣核行,MySQL數(shù)據(jù)庫引擎會在找到一條數(shù)據(jù)后停止搜索牢硅,而不是繼續(xù)往后查少下一條符合記錄的數(shù)據(jù)。

下面的示例芝雪,只是為了找一下是否有“中國”的用戶减余,很明顯,后面的會比前面的更有效率惩系。(請注意佳励,第一條中是Select *,第二條是Select 1)//?沒有效率的:

$r=mysql_query("SELECT?*?FROM?user?WHERE?country?=?'China'");

if(mysql_num_rows($r)>0){

//?...

}

//?有效率的:

$r=mysql_query("SELECT?1?FROM?user?WHERE?country?=?'China'?LIMIT?1");

if(mysql_num_rows($r)>0){

//?...

}

復(fù)制代碼4. 為搜索字段建索引

索引并不一定就是給主鍵或是唯一的字段蛆挫。如果在你的表中赃承,有某個字段你總要會經(jīng)常用來做搜索,那么悴侵,請為其建立索引吧瞧剖。

從上圖你可以看到那個搜索字串 “l(fā)ast_name LIKE ‘a(chǎn)%’”,一個是建了索引可免,一個是沒有索引抓于,性能差了4倍左右。

另外浇借,你應(yīng)該也需要知道什么樣的搜索是不能使用正常的索引的捉撮。例如,當(dāng)你需要在一篇大的文章中搜索一個詞時妇垢,如: “WHERE post_content LIKE ‘%apple%’”巾遭,索引可能是沒有意義的。你可能需要使用MySQL全文索引 或是自己做一個索引(比如說:搜索關(guān)鍵詞或是Tag什么的)

5. 在Join表的時候使用相當(dāng)類型的例闯估,并將其索引

如果你的應(yīng)用程序有很多 JOIN 查詢灼舍,你應(yīng)該確認(rèn)兩個表中Join的字段是被建過索引的。這樣涨薪,MySQL內(nèi)部會啟動為你優(yōu)化Join的SQL語句的機(jī)制骑素。

而且,這些被用來Join的字段刚夺,應(yīng)該是相同的類型的献丑。例如:如果你要把 DECIMAL 字段和一個 INT 字段Join在一起末捣,MySQL就無法使用它們的索引。對于那些STRING類型创橄,還需要有相同的字符集才行塔粒。(兩個表的字符集有可能不一樣)//?在state中查找company

$r=mysql_query("SELECT?company_name?FROM?users

LEFT?JOIN?companies?ON?(users.state?=?companies.state)

WHERE?users.id?=?$user_id");

//?兩個?state?字段應(yīng)該是被建過索引的,而且應(yīng)該是相當(dāng)?shù)念愋涂鹫嗤淖址?/p>

復(fù)制代碼6. 千萬不要 ORDER BY RAND()

想打亂返回的數(shù)據(jù)行卒茬?隨機(jī)挑一個數(shù)據(jù)?真不知道誰發(fā)明了這種用法咖熟,但很多新手很喜歡這樣用圃酵。但你確不了解這樣做有多么可怕的性能問題。

如果你真的想把返回的數(shù)據(jù)行打亂了馍管,你有N種方法可以達(dá)到這個目的郭赐。這樣使用只讓你的數(shù)據(jù)庫的性能呈指數(shù)級的下降。這里的問題是:MySQL會不得 不去執(zhí)行RAND()函數(shù)(很耗CPU時間)确沸,而且這是為了每一行記錄去記行捌锭,然后再對其排序。就算是你用了Limit 1也無濟(jì)于事(因為要排序)

下面的示例是隨機(jī)挑一條記錄//?千萬不要這樣做:

$r=mysql_query("SELECT?username?FROM?user?ORDER?BY?RAND()?LIMIT?1");

//?這要會更好:

$r=mysql_query("SELECT?count(*)?FROM?user");

$d=mysql_fetch_row($r);

$rand=mt_rand(0,$d[0]-1);

$r=mysql_query("SELECT?username?FROM?user?LIMIT?$rand,?1");

復(fù)制代碼7. 避免 SELECT *

從數(shù)據(jù)庫里讀出越多的數(shù)據(jù)罗捎,那么查詢就會變得越慢观谦。并且,如果你的數(shù)據(jù)庫服務(wù)器和WEB服務(wù)器是兩臺獨(dú)立的服務(wù)器的話桨菜,這還會增加網(wǎng)絡(luò)傳輸?shù)呢?fù)載豁状。

所以,你應(yīng)該養(yǎng)成一個需要什么就取什么的好的習(xí)慣倒得。//?不推薦

$r=mysql_query("SELECT?*?FROM?user?WHERE?user_id?=?1");

$d=mysql_fetch_assoc($r);

echo"Welcome?{$d['username']}";

//?推薦

$r=mysql_query("SELECT?username?FROM?user?WHERE?user_id?=?1");

$d=mysql_fetch_assoc($r);

echo"Welcome?{$d['username']}";

復(fù)制代碼8. 永遠(yuǎn)為每張表設(shè)置一個ID

我們應(yīng)該為數(shù)據(jù)庫里的每張表都設(shè)置一個ID做為其主鍵泻红,而且最好的是一個INT型的(推薦使用UNSIGNED),并設(shè)置上自動增加的AUTO_INCREMENT標(biāo)志霞掺。

就算是你 users 表有一個主鍵叫 “email”的字段谊路,你也別讓它成為主鍵。使用 VARCHAR 類型來當(dāng)主鍵會使用得性能下降菩彬。另外缠劝,在你的程序中,你應(yīng)該使用表的ID來構(gòu)造你的數(shù)據(jù)結(jié)構(gòu)挤巡。

而且剩彬,在MySQL數(shù)據(jù)引擎下酷麦,還有一些操作需要使用主鍵矿卑,在這些情況下,主鍵的性能和設(shè)置變得非常重要沃饶,比如母廷,集群轻黑,分區(qū)……

在這里,只有一個情況是例外琴昆,那就是“關(guān)聯(lián)表”的“外鍵”氓鄙,也就是說,這個表的主鍵业舍,通過若干個別的表的主鍵構(gòu)成抖拦。我們把這個情況叫做“外鍵”。比 如:有一個“學(xué)生表”有學(xué)生的ID舷暮,有一個“課程表”有課程ID态罪,那么,“成績表”就是“關(guān)聯(lián)表”了下面,其關(guān)聯(lián)了學(xué)生表和課程表复颈,在成績表中,學(xué)生ID和課 程ID叫“外鍵”其共同組成主鍵沥割。

9. 使用 ENUM 而不是 VARCHAR

ENUM 類型是非澈睦玻快和緊湊的。在實際上机杜,其保存的是 TINYINT帜讲,但其外表上顯示為字符串。這樣一來椒拗,用這個字段來做一些選項列表變得相當(dāng)?shù)耐昝馈?/p>

如果你有一個字段舒帮,比如“性別”,“國家”陡叠,“民族”玩郊,“狀態(tài)”或“部門”,你知道這些字段的取值是有限而且固定的枉阵,那么译红,你應(yīng)該使用 ENUM 而不是 VARCHAR。

MySQL也有一個“建議”(見第十條)告訴你怎么去重新組織你的表結(jié)構(gòu)兴溜。當(dāng)你有一個 VARCHAR 字段時侦厚,這個建議會告訴你把其改成 ENUM 類型。使用 PROCEDURE ANALYSE() 你可以得到相關(guān)的建議拙徽。

10. 從 PROCEDURE ANALYSE() 取得建議

PROCEDURE ANALYSE() 會讓 MySQL 幫你去分析你的字段和其實際的數(shù)據(jù)刨沦,并會給你一些有用的建議。只有表中有實際的數(shù)據(jù)膘怕,這些建議才會變得有用想诅,因為要做一些大的決定是需要有數(shù)據(jù)作為基礎(chǔ)的。

例如,如果你創(chuàng)建了一個 INT 字段作為你的主鍵来破,然而并沒有太多的數(shù)據(jù)篮灼,那么,PROCEDURE ANALYSE()會建議你把這個字段的類型改成 MEDIUMINT 徘禁∽缬眨或是你使用了一個 VARCHAR 字段,因為數(shù)據(jù)不多送朱,你可能會得到一個讓你把它改成 ENUM 的建議娘荡。這些建議,都是可能因為數(shù)據(jù)不夠多驶沼,所以決策做得就不夠準(zhǔn)它改。

在phpmyadmin里,你可以在查看表時商乎,點(diǎn)擊 “Propose table structure” 來查看這些建議

一定要注意央拖,這些只是建議,只有當(dāng)你的表里的數(shù)據(jù)越來越多時鹉戚,這些建議才會變得準(zhǔn)確鲜戒。一定要記住,你才是最終做決定的人抹凳。

11. 盡可能的使用 NOT NULL

除非你有一個很特別的原因去使用 NULL 值遏餐,你應(yīng)該總是讓你的字段保持 NOT NULL。這看起來好像有點(diǎn)爭議赢底,請往下看失都。

首先,問問你自己“Empty”和“NULL”有多大的區(qū)別(如果是INT幸冻,那就是0和NULL)粹庞?如果你覺得它們之間沒有什么區(qū)別,那么你就不要使用NULL洽损。(你知道嗎庞溜?在 Oracle 里,NULL 和 Empty 的字符串是一樣的碑定!)

不要以為 NULL 不需要空間流码,其需要額外的空間,并且延刘,在你進(jìn)行比較的時候漫试,你的程序會更復(fù)雜。 當(dāng)然碘赖,這里并不是說你就不能使用NULL了驾荣,現(xiàn)實情況是很復(fù)雜的外构,依然會有些情況下,你需要使用NULL值秘车。

下面摘自MySQL自己的文檔:“NULL?columnsrequireadditional?spaceinthe?row?to?record?whether?their?values?are?NULL.ForMyISAMtables,each?NULL?column?takes?one?bit?extra,rounded?up?to?the?nearestbyte.”

復(fù)制代碼12. Prepared Statements

Prepared Statements很像存儲過程典勇,是一種運(yùn)行在后臺的SQL語句集合劫哼,我們可以從使用 prepared statements 獲得很多好處叮趴,無論是性能問題還是安全問題。

Prepared Statements 可以檢查一些你綁定好的變量权烧,這樣可以保護(hù)你的程序不會受到“SQL注入式”攻擊眯亦。當(dāng)然,你也可以手動地檢查你的這些變量般码,然而妻率,手動的檢查容易出問題, 而且很經(jīng)常會被程序員忘了板祝。當(dāng)我們使用一些framework或是ORM的時候宫静,這樣的問題會好一些。

在性能方面券时,當(dāng)一個相同的查詢被使用多次的時候孤里,這會為你帶來可觀的性能優(yōu)勢。你可以給這些Prepared Statements定義一些參數(shù)橘洞,而MySQL只會解析一次捌袜。

雖然最新版本的MySQL在傳輸Prepared Statements是使用二進(jìn)制形勢,所以這會使得網(wǎng)絡(luò)傳輸非常有效率炸枣。

當(dāng)然虏等,也有一些情況下,我們需要避免使用Prepared Statements适肠,因為其不支持查詢緩存霍衫。但據(jù)說版本5.1后支持了。

在PHP中要使用prepared statements侯养,你可以查看其使用手冊:mysqli 擴(kuò)展 或是使用數(shù)據(jù)庫抽象層慕淡,如: PDO.//?創(chuàng)建?prepared?statement

if($stmt=$mysqli->prepare("SELECT?username?FROM?user?WHERE?state=?")){

//?綁定參數(shù)

$stmt->bind_param("s",$state);

//?執(zhí)行

$stmt->execute();

//?綁定結(jié)果

$stmt->bind_result($username);

//?移動游標(biāo)

$stmt->fetch();

printf("%s?is?from?%s\n",$username,$state);

$stmt->close();

}

復(fù)制代碼13. 無緩沖的查詢

正常的情況下,當(dāng)你在當(dāng)你在你的腳本中執(zhí)行一個SQL語句的時候沸毁,你的程序會停在那里直到?jīng)]這個SQL語句返回峰髓,然后你的程序再往下繼續(xù)執(zhí)行。你可以使用無緩沖查詢來改變這個行為息尺。

關(guān)于這個事情携兵,在PHP的文檔中有一個非常不錯的說明: mysql_unbuffered_query() 函數(shù):“mysql_unbuffered_query()sends?the?SQL?query?query?toMySQLwithout?automatically?fetchingandbuffering?the?result?rowsasmysql_query()does.Thissaves?a?considerable?amount?of?memorywithSQL?queries?that?produce?large?result?sets,andyou?can?start?working?on?the?resultsetimmediately?after?the?first?row?has?been?retrievedasyou?don’t?have?to?waituntilthe?complete?SQL?query?has?been?performed.”

復(fù)制代碼上面那句話翻譯過來是說,mysql_unbuffered_query() 發(fā)送一個SQL語句到MySQL而并不像mysql_query()一樣去自動fethch和緩存結(jié)果搂誉。這會相當(dāng)節(jié)約很多可觀的內(nèi)存徐紧,尤其是那些會產(chǎn)生大 量結(jié)果的查詢語句,并且,你不需要等到所有的結(jié)果都返回并级,只需要第一行數(shù)據(jù)返回的時候拂檩,你就可以開始馬上開始工作于查詢結(jié)果了。

然而嘲碧,這會有一些限制稻励。因為你要么把所有行都讀走,或是你要在進(jìn)行下一次的查詢前調(diào)用 mysql_free_result() 清除結(jié)果愈涩。而且望抽, mysql_num_rows() 或 mysql_data_seek() 將無法使用。所以履婉,是否使用無緩沖的查詢你需要仔細(xì)考慮煤篙。

14. 把IP地址存成 UNSIGNED INT

很多程序員都會創(chuàng)建一個 VARCHAR(15) 字段來存放字符串形式的IP而不是整形的IP。如果你用整形來存放毁腿,只需要4個字節(jié)辑奈,并且你可以有定長的字段。而且已烤,這會為你帶來查詢上的優(yōu)勢鸠窗,尤其是當(dāng) 你需要使用這樣的WHERE條件:IP between ip1 and ip2。

我們必需要使用UNSIGNED INT草戈,因為 IP地址會使用整個32位的無符號整形塌鸯。

而你的查詢,你可以使用 INET_ATON() 來把一個字符串IP轉(zhuǎn)成一個整形唐片,并使用 INET_NTOA() 把一個整形轉(zhuǎn)成一個字符串IP丙猬。在PHP中,也有這樣的函數(shù) ip2long() 和 long2ip()费韭。$r="UPDATE?users?SET?ip?=?INET_ATON('{$_SERVER['REMOTE_ADDR']}')?WHERE?user_id?=?$user_id";

復(fù)制代碼15. 固定長度的表會更快

如果表中的所有字段都是“固定長度”的茧球,整個表會被認(rèn)為是 “static” 或 “fixed-length”。 例如星持,表中沒有如下類型的字段: VARCHAR抢埋,TEXT,BLOB督暂。只要你包括了其中一個這些字段揪垄,那么這個表就不是“固定長度靜態(tài)表”了,這樣逻翁,MySQL 引擎會用另一種方法來處理饥努。

固定長度的表會提高性能,因為MySQL搜尋得會更快一些八回,因為這些固定的長度是很容易計算下一個數(shù)據(jù)的偏移量的酷愧,所以讀取的自然也會很快驾诈。而如果字段不是定長的,那么溶浴,每一次要找下一條的話乍迄,需要程序找到主鍵。

并且士败,固定長度的表也更容易被緩存和重建闯两。不過,唯一的副作用是拱烁,固定長度的字段會浪費(fèi)一些空間生蚁,因為定長的字段無論你用不用噩翠,他都是要分配那么多的空間戏自。

使用“垂直分割”技術(shù)(見下一條),你可以分割你的表成為兩個一個是定長的伤锚,一個則是不定長的擅笔。

16. 垂直分割

“垂直分割”是一種把數(shù)據(jù)庫中的表按列變成幾張表的方法,這樣可以降低表的復(fù)雜度和字段的數(shù)目屯援,從而達(dá)到優(yōu)化的目的猛们。(以前,在銀行做過項目狞洋,見過一張表有100多個字段弯淘,很恐怖)

示例一:在Users表中有一個字段是家庭地址,這個字段是可選字段吉懊,相比起庐橙,而且你在數(shù)據(jù)庫操作的時候除了個 人信息外,你并不需要經(jīng)常讀取或是改寫這個字段借嗽。那么态鳖,為什么不把他放到另外一張表中呢? 這樣會讓你的表有更好的性能恶导,大家想想是不是浆竭,大量的時候,我對于用戶表來說惨寿,只有用戶ID邦泄,用戶名,口令裂垦,用戶角色等會被經(jīng)常使用顺囊。小一點(diǎn)的表總是會有 好的性能。

示例二: 你有一個叫 “l(fā)ast_login” 的字段缸废,它會在每次用戶登錄時被更新包蓝。但是驶社,每次更新時會導(dǎo)致該表的查詢緩存被清空。所以测萎,你可以把這個字段放到另一個表中亡电,這樣就不會影響你對用戶ID,用戶名硅瞧,用戶角色的不停地讀取了份乒,因為查詢緩存會幫你增加很多性能。

另外腕唧,你需要注意的是或辖,這些被分出去的字段所形成的表,你不會經(jīng)常性地去Join他們枣接,不然的話颂暇,這樣的性能會比不分割時還要差,而且但惶,會是極數(shù)級的下降耳鸯。

17. 拆分大的 DELETE 或 INSERT 語句

如果你需要在一個在線的網(wǎng)站上去執(zhí)行一個大的 DELETE 或 INSERT 查詢,你需要非常小心膀曾,要避免你的操作讓你的整個網(wǎng)站停止相應(yīng)县爬。因為這兩個操作是會鎖表的,表一鎖住了添谊,別的操作都進(jìn)不來了财喳。

Apache 會有很多的子進(jìn)程或線程。所以斩狱,其工作起來相當(dāng)有效率耳高,而我們的服務(wù)器也不希望有太多的子進(jìn)程,線程和數(shù)據(jù)庫鏈接喊废,這是極大的占服務(wù)器資源的事情祝高,尤其是內(nèi)存。

如果你把你的表鎖上一段時間污筷,比如30秒鐘工闺,那么對于一個有很高訪問量的站點(diǎn)來說,這30秒所積累的訪問進(jìn)程/線程瓣蛀,數(shù)據(jù)庫鏈接陆蟆,打開的文件數(shù),可能不僅僅會讓你泊WEB服務(wù)Crash惋增,還可能會讓你的整臺服務(wù)器馬上掛了叠殷。

所以,如果你有一個大的處理诈皿,你定你一定把其拆分林束,使用 LIMIT 條件是一個好的方法像棘。下面是一個示例:while(1){

//每次只做1000條

mysql_query("DELETE?FROM?logs?WHERE?log_date?<=?'2009-11-01'?LIMIT?1000");

if(mysql_affected_rows()==0){

//?沒得可刪了,退出壶冒!

break;

}

//?每次都要休息一會兒

usleep(50000);

}

復(fù)制代碼18. 越小的列會越快

對于大多數(shù)的數(shù)據(jù)庫引擎來說缕题,硬盤操作可能是最重大的瓶頸。所以胖腾,把你的數(shù)據(jù)變得緊湊會對這種情況非常有幫助烟零,因為這減少了對硬盤的訪問。

參看 MySQL 的文檔 Storage Requirements 查看所有的數(shù)據(jù)類型咸作。

如果一個表只會有幾列罷了(比如說字典表锨阿,配置表),那么记罚,我們就沒有理由使用 INT 來做主鍵墅诡,使用 MEDIUMINT, SMALLINT 或是更小的 TINYINT 會更經(jīng)濟(jì)一些。如果你不需要記錄時間毫胜,使用 DATE 要比 DATETIME 好得多书斜。

當(dāng)然诬辈,你也需要留夠足夠的擴(kuò)展空間酵使,不然,你日后來干這個事焙糟,你會死的很難看口渔,參看Slashdot的例子(2009年11月06日),一個簡單的ALTER TABLE語句花了3個多小時穿撮,因為里面有一千六百萬條數(shù)據(jù)缺脉。

19. 選擇正確的存儲引擎

在 MySQL 中有兩個存儲引擎 MyISAM 和 InnoDB,每個引擎都有利有弊悦穿」ダ瘢酷殼以前文章《MySQL: InnoDB 還是 MyISAM?》討論和這個事情。

MyISAM 適合于一些需要大量查詢的應(yīng)用栗柒,但其對于有大量寫操作并不是很好礁扮。甚至你只是需要update一個字段,整個表都會被鎖起來瞬沦,而別的進(jìn)程太伊,就算是讀進(jìn)程都 無法操作直到讀操作完成。另外逛钻,MyISAM 對于 SELECT COUNT(*) 這類的計算是超快無比的僚焦。

InnoDB 的趨勢會是一個非常復(fù)雜的存儲引擎,對于一些小的應(yīng)用曙痘,它會比 MyISAM 還慢芳悲。他是它支持“行鎖” 立肘,于是在寫操作比較多的時候,會更優(yōu)秀名扛。并且赛不,他還支持更多的高級應(yīng)用,比如:事務(wù)罢洲。

下面是MySQL的手冊

target=”_blank”MyISAM Storage Engine

InnoDB Storage Engine

20. 使用一個對象關(guān)系映射器(Object Relational Mapper)

使用 ORM (Object Relational Mapper)踢故,你能夠獲得可靠的性能增漲。一個ORM可以做的所有事情惹苗,也能被手動的編寫出來殿较。但是,這需要一個高級專家桩蓉。

ORM 的最重要的是“Lazy Loading”淋纲,也就是說,只有在需要的去取值的時候才會去真正的去做院究。但你也需要小心這種機(jī)制的副作用洽瞬,因為這很有可能會因為要去創(chuàng)建很多很多小的查詢反而會降低性能。

ORM 還可以把你的SQL語句打包成一個事務(wù)业汰,這會比單獨(dú)執(zhí)行他們快得多得多伙窃。

目前,個人最喜歡的PHP的ORM是:Doctrine样漆。

21. 小心“永久鏈接”

“永久鏈接”的目的是用來減少重新創(chuàng)建MySQL鏈接的次數(shù)为障。當(dāng)一個鏈接被創(chuàng)建了,它會永遠(yuǎn)處在連接的狀態(tài)放祟,就算是數(shù)據(jù)庫操作已經(jīng)結(jié)束了鳍怨。而且,自 從我們的Apache開始重用它的子進(jìn)程后——也就是說跪妥,下一次的HTTP請求會重用Apache的子進(jìn)程鞋喇,并重用相同的 MySQL 鏈接。

PHP手冊:mysql_pconnect()

在理論上來說眉撵,這聽起來非常的不錯侦香。但是從個人經(jīng)驗(也是大多數(shù)人的)上來說,這個功能制造出來的麻煩事更多执桌。因為鄙皇,你只有有限的鏈接數(shù),內(nèi)存問題仰挣,文件句柄數(shù)伴逸,等等。

而且膘壶,Apache 運(yùn)行在極端并行的環(huán)境中错蝴,會創(chuàng)建很多很多的了進(jìn)程洲愤。這就是為什么這種“永久鏈接”的機(jī)制工作地不好的原因。在你決定要使用“永久鏈接”之前顷锰,你需要好好地考慮一下你的整個系統(tǒng)的架構(gòu)柬赐。

轉(zhuǎn)載自:酷殼網(wǎng)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市官紫,隨后出現(xiàn)的幾起案子肛宋,更是在濱河造成了極大的恐慌,老刑警劉巖束世,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件酝陈,死亡現(xiàn)場離奇詭異,居然都是意外死亡毁涉,警方通過查閱死者的電腦和手機(jī)沉帮,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來贫堰,“玉大人穆壕,你說我怎么就攤上這事∑淦粒” “怎么了喇勋?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長漫玄。 經(jīng)常有香客問我茄蚯,道長,這世上最難降的妖魔是什么睦优? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮壮不,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己幌羞,他們只是感情好是晨,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著健蕊,像睡著了一般菱阵。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上缩功,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天晴及,我揣著相機(jī)與錄音,去河邊找鬼嫡锌。 笑死虑稼,一個胖子當(dāng)著我的面吹牛琳钉,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蛛倦,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼歌懒,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了溯壶?” 一聲冷哼從身側(cè)響起及皂,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎且改,沒想到半個月后躲庄,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡钾虐,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年噪窘,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片效扫。...
    茶點(diǎn)故事閱讀 39,711評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡倔监,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出菌仁,到底是詐尸還是另有隱情浩习,我是刑警寧澤,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布济丘,位于F島的核電站谱秽,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏摹迷。R本人自食惡果不足惜疟赊,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望峡碉。 院中可真熱鬧近哟,春花似錦、人聲如沸鲫寄。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽地来。三九已至戳玫,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間未斑,已是汗流浹背咕宿。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人荠列。 一個月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓类浪,卻偏偏與公主長得像,于是被迫代替她去往敵國和親肌似。 傳聞我的和親對象是個殘疾皇子费就,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評論 2 353

推薦閱讀更多精彩內(nèi)容