公司之前購買過一個源碼,寫這個源碼的工程師當時沒有考慮全面,設計數(shù)據(jù)庫表結構有點問題,當公司的數(shù)據(jù)達到幾十萬級別的時候,該網(wǎng)站基本就跑不動了。原因是查詢的時候沒有使用索引,造成了大量的數(shù)據(jù)庫慢查詢,服務器同時存在許多php-fpm進程運行。幾乎耗盡了服務器cpu和內(nèi)存。
當定位到問題后,重新設計了表結構,給對應的幾個字段加上了索引。加上索引之后,還是偶爾會出現(xiàn)cpu、內(nèi)存快耗盡的情況。對于這種情況,我編寫了一個shell腳本,用來監(jiān)控服務器內(nèi)存的使用率,一旦達到預設值時,殺死所有的php-fpm進程,釋放服務器的壓力。
首先,我們要獲取服務器的內(nèi)存使用率。通過free可以獲取總內(nèi)存大小以及使用內(nèi)存多少
#?free? ??????????????total????????used????????free??????shared??buff/cache???available Mem:????????7999972?????5432684??????152496????????2480?????2414792?????2284544 Swap:?????????????0???????????0???????????0
我們需要的是Mem那行的,total以及used項。我們通過grep以及awk命令,可以獲得當前系統(tǒng)的內(nèi)存使用率。
free?|grep?-i?mem?|?awk?'{n=$3/$2;?printf("%.0f",?n?*?100)}'
獲得了內(nèi)存使用率后,然后拿它和預設值作比較。當大于預設值時,就把系統(tǒng)中所以php-fpm進程殺死。那么接下來的工作就是如何找出系統(tǒng)中所有php-fpm進程,以及如何去殺死這些進程。
立即學習“PHP免費學習筆記(深入)”;
想要獲取系統(tǒng)所有的php-fpm進程,可以使用ps命令,然后結合grep過濾即可。
ps?aux?|?grep?php-fpm?|?grep?-v?grep?|?grep?-v?master www??????21210??0.0??0.1?157852??8596??????????S????19:33???0:00?php-fpm:?pool?www www??????21211??0.0??0.1?157852??8596??????????S????19:33???0:00?php-fpm:?pool?www ……
通過上面的命令獲取到了所有php-fpm進程,然后我們遍歷這些信息,通過kill 進程號,來殺死php-fpm進程。
下面給出完整的shell腳本程序:
#!/bin/bash #?內(nèi)存檢查,超過70%,殺死所有php-fpm進程 MEM_LIM=70 used=$(free?|?grep?-i?mem?|?awk?'{n=$3/$2;printf("%.0f",?n*100)}') if?((used?>?MEM_LIM));then ????pids=`ps?aux?|?grep?php-fpm?|?grep?-v?grep?|?grep?-v?master?|?awk?'{print?$2}'` ????for?pid?in?$pids ????do ???????kill?-9?$pid ????done fi
上述的腳本是非常簡單了,清晰命令。下面總結下該腳本程序使用了哪些知識點:
-
free命令獲取內(nèi)存使用率
-
ps命令獲取所有php-fpm進程
-
kill命令殺死進程
-
shell編程條件分支以及循環(huán)結構