mysql并發(fā)查詢慢,mysql數據庫cpu飆升800?
mysql中CPU占用過高的診斷思路,舉個栗子~
mpstat -P ALL 1,查看cpu使用情況,主要消耗在sys即os系統(tǒng)調用上
perf top,cpu主要消耗在_spin_lock
生成perf report查看詳細情況
CPU主要消耗在mutex爭用上,說明有鎖熱點。
采用pt-pmp跟蹤mysqld執(zhí)行情況,熱點主要集中在mem_heap_alloc和mem_heap_free上。
Pstack提供更詳細的API調用棧
#0 0x0000003e0caf80cf in __lll_unlock_wake_private () from /lib64/libc.so.6#1 0x0000003e0ca7cf6a in _L_unlock_5936 () from /lib64/libc.so.6#2 0x0000003e0ca78bbc in _int_free () from /lib64/libc.so.6#3 0x000000000097dcb3 in mem_area_free(void*, mem_pool_t*) ()#4 0x000000000097d2d2 in mem_heap_block_free(mem_block_info_t*, mem_block_info_t*) ()#5 0x00000000009e6474 in row_vers_build_for_consistent_read(unsigned char const*, mtr_t*, dict_index_t*, unsigned long**, read_view_t*, mem_block_info_t**, mem_block_info_t*, unsigned char**) ()#6 0x00000000009dce75 in row_search_for_mysql(unsigned char*, unsigned long, row_prebuilt_t*, unsigned long, unsigned long) ()#7 0x0000000000939c95 in ha_innobase::index_read(unsigned char*, unsigned char const*, unsigned int, ha_rkey_function) ()Innodb在讀取數據記錄時的API路徑為
row_search_for_mysql --》row_vers_build_for_consistent_read --》mem_heap_create_block_func --》mem_area_alloc --》malloc --》_L_unlock_10151 --》__lll_unlock_wait_privaterow_vers_build_for_consistent_read會陷入一個死循環(huán),跳出條件是該條記錄不需要快照讀或者已經從undo中找出對應的快照版本,每次循環(huán)都會調用mem_heap_alloc/free。
而該表的記錄更改很頻繁,導致其undo history list比較長,搜索快照版本的代價更大,就會頻繁的申請和釋放堆內存。
Linux原生的內存庫函數為ptmalloc,malloc/free調用過多時很容易產生鎖熱點。
當多條 SQL 并發(fā)執(zhí)行時,會最終觸發(fā)os層面的spinlock,導致上述情形。
解決方案
將mysqld的內存庫函數替換成tcmalloc,相比ptmalloc,tcmalloc可以更好的支持高并發(fā)調用。
修改my.cnf,添加如下參數并重啟
[mysqld_safe]malloc-lib=tcmalloc上周五早上7點執(zhí)行的操作,到現在超過72小時,期間該實例沒有再出現cpu長期飆高的情形。
以下是修改前后cpu使用率對比