MySQL占用CPU過(guò)高,排查原因及解決方案
一、mysql中的wait_timeout坑
mysql> show variables like '%timeout%';
首先你要明白:
wait_timeout
— 指的是mysql在關(guān)閉一個(gè)非交互的連接之前所要等待的秒數(shù),其取值范圍為1-2147483(Windows),1-31536000(linux),默認(rèn)值28800。nteractive_time
— 指的是mysql在關(guān)閉一個(gè)交互的連接之前所要等待的秒數(shù)(交互連接如mysql gui tool中的連接),其取值范圍隨wait_timeout變動(dòng),默認(rèn)值28800。
所謂的交互式連接,即在mysql_real_connect()函數(shù)中使用了CLIENT_INTERACTIVE選項(xiàng)。說(shuō)得直白一點(diǎn),通過(guò)mysql客戶端連接數(shù)據(jù)庫(kù)是交互式連接,通過(guò)jdbc連接數(shù)據(jù)庫(kù)是非交互式連接。
MySQL 的默認(rèn)設(shè)置下,當(dāng)一個(gè)連接的空閑時(shí)間超過(guò)8小時(shí)后,一到高峰期肯定會(huì)造成,會(huì)有太多的TCP連接沒(méi)關(guān)閉,數(shù)據(jù)庫(kù)連接數(shù)肯定是不夠。
從而會(huì)產(chǎn)生CPU占用過(guò)高,服務(wù)器告警等問(wèn)題。因EPG的一個(gè)訪問(wèn)一次對(duì)數(shù)據(jù)庫(kù)操作量不大,查詢完數(shù)據(jù)就完成ok了,wait_timeout 設(shè)置在120s內(nèi)就很完美了。
1、第一種修改方式 需重啟MySQL(6.5為例)
1.1、修改參數(shù)配置文件
vi /etc/my.cnf [mysqld] wait_timeout?? ?= 90 interactive_timeout?? ?=? 90 #增加以上兩列即可,因?yàn)楣俜轿臋n要求修改此參數(shù)必須同時(shí)修改interactive_timeout
1.2、重啟數(shù)據(jù)庫(kù)
service mysqld restart?
或者
/etc/init.d/mysql restart
1.3、查看數(shù)據(jù)庫(kù)參數(shù)是否修改成功
連接MySQL 然后查看
show variables like ‘wait_timeout';
2、第二種修改方式 不需重啟MySQL(數(shù)據(jù)庫(kù)重啟就失效)
mysql>?show?variables?like?'%timeout'; +-------------------------+-------+ |?Variable_name???????????|?Value?| +-------------------------+-------+ |?connect_timeout?????????|?10????| |?delayed_insert_timeout??|?300???| |?interactive_timeout?????|?200???| |?net_read_timeout????????|?30????| |?net_write_timeout???????|?60????| |?slave_net_timeout???????|?3600??| |?table_lock_wait_timeout?|?50????| |?wait_timeout????????????|?200???| +-------------------------+-------+ mysql>?set?global?interactive_timeout=100; Query?OK,?0?rows?affected?(0.00?sec)
退出后重新登錄mysql
如果修改interactive_timeout的話wait_timeout也會(huì)跟著變,而只修改wait_timeout是不生效的。
二、排查有沒(méi)有地方占用SQL資源過(guò)多
show databases;
3、use 數(shù)據(jù)庫(kù)名; #切換到常用數(shù)據(jù)庫(kù)
4、show full processlist; #顯示哪些線程正在運(yùn)行
或者:
-- select * from information_schema.PROCESSLIST?where info is not null;
說(shuō)明各列的含義和用途
- id列: 一個(gè)標(biāo)識(shí),你要kill 一個(gè)語(yǔ)句的時(shí)候很有用。
- user列: 顯示當(dāng)前用戶,如果不是root,這個(gè)命令就只顯示你權(quán)限范圍內(nèi)的sql語(yǔ)句。
- host列: 顯示這個(gè)語(yǔ)句是從哪個(gè)ip 的哪個(gè)端口上發(fā)出的??捎脕?lái)追蹤出問(wèn)題語(yǔ)句的用戶。
- db列: 顯示這個(gè)進(jìn)程目前連接的是哪個(gè)數(shù)據(jù)庫(kù)。
- command列: 顯示當(dāng)前連接的執(zhí)行的命令,一般就是休眠(sleep),查詢(query),連接(connect)。
通常代表資源未釋放,如果是通過(guò)連接池,sleep狀態(tài)應(yīng)該恒定在一定數(shù)量范圍內(nèi)
實(shí)戰(zhàn)范例:因前端數(shù)據(jù)輸出時(shí)(特別是輸出到用戶終端)未及時(shí)關(guān)閉數(shù)據(jù)庫(kù)連接,導(dǎo)致因網(wǎng)絡(luò)連接速度產(chǎn)生大量sleep連接,在網(wǎng)速出現(xiàn)異常時(shí),數(shù)據(jù)庫(kù)too many connections掛死。
簡(jiǎn)單解讀,數(shù)據(jù)查詢和執(zhí)行通常只需要不到0.01秒,而網(wǎng)絡(luò)輸出通常需要1秒左右甚至更長(zhǎng),原本數(shù)據(jù)連接在0.01秒即可釋放,但是因?yàn)榍岸顺绦蛭磮?zhí)行close操作,直接輸出結(jié)果,那么在結(jié)果未展現(xiàn)在用戶桌面前,該數(shù)據(jù)庫(kù)連接一直維持在sleep狀態(tài)!
- time列: 此這個(gè)狀態(tài)持續(xù)的時(shí)間,單位是秒。
- state列: 顯示使用當(dāng)前連接的sql語(yǔ)句的狀態(tài),很重要的列,后續(xù)會(huì)有所有的狀態(tài)的描述,請(qǐng)注意,state只是語(yǔ)句執(zhí)行中的某一個(gè)狀態(tài),一個(gè)sql語(yǔ)句,已查詢?yōu)槔?,可能需要?jīng)過(guò)copying to tmp table,Sorting result,Sending data等狀態(tài)才可以完成。
- info列: 顯示這個(gè)sql語(yǔ)句,因?yàn)殚L(zhǎng)度有限,所以長(zhǎng)的sql語(yǔ)句就顯示不全,但是一個(gè)判斷問(wèn)題語(yǔ)句的重要依據(jù)。
其中state的狀態(tài)十分關(guān)鍵
下表列出state主要狀態(tài)和描述:
狀態(tài) | 描述 |
Checking table | 正在檢查數(shù)據(jù)表(這是自動(dòng)的)。 |
Closing tables | 正在將表中修改的數(shù)據(jù)刷新到磁盤中,同時(shí)正在關(guān)閉已經(jīng)用完的表。這是一個(gè)很快的操作,如果不是這樣的話,就應(yīng)該確認(rèn)磁盤空間是否已經(jīng)滿了或者磁盤是否正處于重負(fù)中。 |
Connect Out | 復(fù)制從服務(wù)器正在連接主服務(wù)器。 |
Copying to tmp table on disk | 由于臨時(shí)結(jié)果集大于tmp_table_size,正在將臨時(shí)表從內(nèi)存存儲(chǔ)轉(zhuǎn)為磁盤存儲(chǔ)以此節(jié)省內(nèi)存。 |
Creating tmp table | 正在創(chuàng)建臨時(shí)表以存放部分查詢結(jié)果。 |
deleting from main table | 服務(wù)器正在執(zhí)行多表刪除中的第一部分,剛刪除第一個(gè)表。 |
deleting from reference tables | 服務(wù)器正在執(zhí)行多表刪除中的第二部分,正在刪除其他表的記錄。 |
Flushing tables | 正在執(zhí)行FLUSH TABLES,等待其他線程關(guān)閉數(shù)據(jù)表。 |
Killed | 發(fā)送了一個(gè)kill請(qǐng)求給某線程,那么這個(gè)線程將會(huì)檢查kill標(biāo)志位,同時(shí)會(huì)放棄下一個(gè)kill請(qǐng)求。MySQL會(huì)在每次的主循環(huán)中檢查kill標(biāo)志位,不過(guò)有些情況下該線程可能會(huì)過(guò)一小段才能死掉。如果該線程程被其他線程鎖住了,那么kill請(qǐng)求會(huì)在鎖釋放時(shí)馬上生效。 |
Locked | 被其他查詢鎖住了。 |
Sending data | 正在處理SELECT查詢的記錄,同時(shí)正在把結(jié)果發(fā)送給客戶端。 |
Sorting for group | 正在為GROUP BY做排序。 |
Sorting for order | 正在為ORDER BY做排序。 |
Opening tables | 這個(gè)過(guò)程應(yīng)該會(huì)很快,除非受到其他因素的干擾。例如,在執(zhí)ALTER TABLE或LOCK TABLE語(yǔ)句行完以前,數(shù)據(jù)表無(wú)法被其他線程打開(kāi)。正嘗試打開(kāi)一個(gè)表。 |
Removing duplicates | 正在執(zhí)行一個(gè)SELECT DISTINCT方式的查詢,但是MySQL無(wú)法在前一個(gè)階段優(yōu)化掉那些重復(fù)的記錄。因此,MySQL需要再次去掉重復(fù)的記錄,然后再把結(jié)果發(fā)送給客戶端。 |
Reopen table | 獲得了對(duì)一個(gè)表的鎖,但是必須在表結(jié)構(gòu)修改之后才能獲得這個(gè)鎖。已經(jīng)釋放鎖,關(guān)閉數(shù)據(jù)表,正嘗試重新打開(kāi)數(shù)據(jù)表。 |
Repair by sorting | 修復(fù)指令正在排序以創(chuàng)建索引。 |
Repair with keycache | 修復(fù)指令正在利用索引緩存一個(gè)一個(gè)地創(chuàng)建新索引。它會(huì)比Repair by sorting慢些。 |
Searching rows for update | 正在講符合條件的記錄找出來(lái)以備更新。它必須在UPDATE要修改相關(guān)的記錄之前就完成了。 |
Sleeping | 正在等待客戶端發(fā)送新請(qǐng)求. |
System lock | 正在等待取得一個(gè)外部的系統(tǒng)鎖。如果當(dāng)前沒(méi)有運(yùn)行多個(gè)mysqld服務(wù)器同時(shí)請(qǐng)求同一個(gè)表,那么可以通過(guò)增加--skip-external-locking參數(shù)來(lái)禁止外部系統(tǒng)鎖。 |
Upgrading lock | INSERT DELAYED正在嘗試取得一個(gè)鎖表以插入新記錄。 |
Updating | 正在搜索匹配的記錄,并且修改它們。 |
User Lock | 正在等待GET_LOCK()。 |
Waiting for tables | 該線程得到通知,數(shù)據(jù)表結(jié)構(gòu)已經(jīng)被修改了,需要重新打開(kāi)數(shù)據(jù)表以取得新的結(jié)構(gòu)。然后,為了能的重新打開(kāi)數(shù)據(jù)表,必須等到所有其他線程關(guān)閉這個(gè)表。以下幾種情況下會(huì)產(chǎn)生這個(gè)通知:FLUSH TABLES tbl_name, ALTER TABLE, RENAME TABLE, REPAIR TABLE, ANALYZE TABLE,或OPTIMIZE TABLE。 |
waiting for handler insert | INSERT DELAYED已經(jīng)處理完了所有待處理的插入操作,正在等待新的請(qǐng)求。 |
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
通過(guò)SQL語(yǔ)句來(lái)備份,還原數(shù)據(jù)庫(kù)
這里僅僅用到了一種方式而已,把數(shù)據(jù)庫(kù)文件備份到磁盤然后在恢復(fù).2010-02-02與MSSQL對(duì)比學(xué)習(xí)MYSQL的心得(一)--基本語(yǔ)法
最開(kāi)始接觸的數(shù)據(jù)庫(kù)為MSSQL,不過(guò)最近項(xiàng)目需求,仔細(xì)學(xué)習(xí)了下MYSQL,下面就對(duì)比MSSQL,把MYSQL的學(xué)習(xí)心得分享給大家2014-06-06mysql存儲(chǔ)過(guò)程之錯(cuò)誤處理實(shí)例詳解
這篇文章主要介紹了mysql存儲(chǔ)過(guò)程之錯(cuò)誤處理,結(jié)合實(shí)例形式詳細(xì)分析了mysql存儲(chǔ)過(guò)程錯(cuò)誤處理相關(guān)原理、操作技巧與注意事項(xiàng),需要的朋友可以參考下2019-12-12mysql中Innodb 行鎖實(shí)現(xiàn)原理
InnoDB的行鎖是通過(guò)索引項(xiàng)加鎖實(shí)現(xiàn)的,分為使用索引和非索引字段檢索的情況,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-10-10關(guān)于Mysql插入中文字符報(bào)錯(cuò)ERROR 1366(HY000)的解決方法
這篇文章主要介紹了關(guān)于Mysql插入中文字符報(bào)錯(cuò)ERROR 1366(HY000)的解決方法,在我們?nèi)粘J褂胢ysql的過(guò)程中會(huì)經(jīng)常遇到各種報(bào)錯(cuò),今天我們就來(lái)看一下ERROR 1366報(bào)錯(cuò)的解決方法吧2023-07-07關(guān)于Mysql查詢帶單引號(hào)及插入帶單引號(hào)字符串問(wèn)題
本文主要介紹的是用mysql_real_escape_string對(duì)用戶提交的表單數(shù)據(jù)進(jìn)行轉(zhuǎn)義處理和通過(guò)addslashes以及mysql_escape_string這3個(gè)類似功能的函數(shù)用法區(qū)別2013-04-04