MySQL性能瓶颈原因
硬件、系统因素
CPU
磁盘I/O
网络性能
操作系统争用
MySQL相关因素
数据库设计
索引、数据类型
应用程序性能
特定请求、短时事务
配置变量
缓冲区、高速缓存、InnoDB
瓶颈分析工具
sysstat工具集
sar、iostat、pidstat、mpstat
perf top
实时显示系统/进程的性能统计信息
pt-ioprofile
MySQL中的iotop
pstack
当MySQL里有线程hang住时,利用pstack排查由于哪些函数调用存在问题
硬件优化
BIOS配置优化
CPU设置最大性能模式,关闭C1E,C-stats
内存设置最大性能模式
关闭NUMA(并关注innodb_numa_interleave选项)
RAID配置优化
RAID-10
CACHE & BBU
WB & FORCE WB (WB是好的,WT是不好的)
使用PCIe-SSD等高速I/O设备
$ numactl --hardware
available: 1 nodes (0)
node 0 cpus: 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39
node 0 size: 65415 MB
node 0 free: 4086 MB
node distances:
node 0
0: 10
available大于1说明numa没关闭
系统优化
io scheduler
deadline(机械盘)/noop(SSD盘)
filesystems
xfs(强烈建议)
ext4(备选)
kernel
vm.swappiness=1~5 (rhel7以后,设置0要慎重)
vm.dirty_background_radio=5
vm.dirty_radio=10
MySQL配置优化
innodb_buffer_pool_size
innodb_log_file_size >= 1G & innodb_log_files_in_group >= 3
innodb_flush_log_at_trx_commit & sync_binlog = ?
innodb_max_dirty_pages_pct
innodb_thread_concurrency = 0
innodb_io_capacity & innodb_io_capacity_max
innodb_max_purge_lag = 0
innodb_autoinc_lock_mode = 2 & binlog_format = row
max_binlog_cache_size
sort/join/read/read rrd buffer、tmp/heap table (session级别的内存不要设置太高)
thread_handling = one-thread-per-connection | pool-of-thread
long_query_time = 0.05
log_error_verbosity = 3
time_zone = "+8:00" (默认值SYSTEM,设置为具体时区即可)
open_files_limit = 65535
back_log = (max_connections * 2)
max_execution_time = 10000 (/*+ MAX_EXECUTION_TIME(10000)*/)
SCHEMA设计优化
核心宗旨
表越窄越好 - 数据类型够用就好,保持高效
表越小越好 - 数据冷热分离,只留最新热数据
查询够高效 - 令每个SQL效率足够高,不阻塞
InnoDB使用
INT AUTO_INC PK for InnoDB
INT unsigned for IPV4, not CHAR(15)
TINYINT for ENUM type
TEXT、BLOB,or in compressed,并且尽量拆分到子表
JOIN列类型(包括长度、字符集)保持一致
多表JOIN时,排序列一定要属于驱动表,否则有额外filesort
select ... from a straight_join b where ... order by b.x; 需要额外的filesort
所有列设置NOT NULL
主键/唯一索引性能优于普通索引
联合索引比普通索引更合适
基数(cardinality)小的列不建独立索引,因为不支持bitmap索引
长字段使用部分索引,而非全部
可以利用冗余反向字段用于反向检索
sphinx等代替like '%xx%'搜索
SQL优化
多用简单SQL
少用子查询(不熟悉子查询时)
少用复杂表JOIN,并且注意JOIN方式
WHERE条件中不使用函数、表达式(虚拟列变通)
注意类型隐式转换
固定业务逻辑封装成存储过程(存储过程可能会影响未来分库分表、读写分离架构)
限制结果集大小
分页改成基于主键的JOIN或子查询
动态改造成静态,或者降低更新频率
从产品端消减不必要的业务
逻辑优化
以下场景,都可以在redis或cache中暂存,最后再定时持久化存储到DB
游戏人员移动,或是一些技能背包字段的设计,更新可以做合并更新等
过节时人们的发短信或是祝福行为
文章被访问每次加1
用户登录每次加1
MySQL性能瓶颈定位
优化耗时/逻辑读 top sql
找到耗时最慢的 top sql (PFS.events_statements_history_long)
排序TIMER_WAIT/ROWS_EXAMINED/ROWS_SENT
定位这些SQL的性能瓶颈(PFS.events_stages_history_long/profiling)
优化物理逻辑 i/o top sql
找到逻辑 i/o 请求最多的对象 (PFS.table_io_waits_summary_by_table)
找到物理 i/o 请求最多的对象 (sys.io_global_by_file_by_bytes)
定位、优化这些请求
不建议打开pfs
几个案例
insert慢的几种场景
innodb_thread_concurrency > 0
innodb_max_purge_lag > 0
innodb_io_capacity太小
redo & binlog 2PC
锁等待
semi-sync延迟
disk full
中间件性能太差,导致mysqld server端大量请求被阻塞
SQL瓶颈分析之PROFILE
mysql> set profiling=1;
mysql> select * from customer order by address_id;
mysql> show profiles;
mysql> show profile for query 1;
SQL效率对比分析之SHOW STATUS
mysql> flush status;
mysql> select * from customer order by address_id;
mysql> show status like 'handler_read_%';
内存临时表
基于内存的临时表(in-memory internal temporary tables),引擎由默认的memory改成了新的TempTable引擎
默认内存相对于tmp-table-size,从16M提升到1G
对varchar、varbinary的处理效率更高
internal_tmp_mem_storage_engine = TempTable | MEMORY
设置变量temptable_max_ram 来控制实际存储内存区域大小,默认为1G
slow query log
解析:
mysqldumpslow
pt-query-digest(推荐)
也可以启用general log + pt-query-digest
可视化管理:Box Anemometer/Query-Digest-UI
相关选项:
long-query-time = 0.01
log_queries_not_using_indexes = 1
log_throttle_queries_not_using_indexes = 60
min_examined_row_limit = 100
log_slow_admin_statements = 1
log_slow_slave_statements = 1
5.7 & 8.0 版本下,可以利用sys schema.statement_analysis视图
执行次数最多的TOP 10 SQL
select ... order by exec_count desc limit 10
平均响应耗时最慢的TOP 10 SQL
select ... order by avg_latency desc limit 10
每次扫描行数最多的TOP 10 SQL
select ... order by rows_examined desc limit 10
优先级高:执行次数多,高峰期慢SQL,核心业务慢SQL
优先级低:执行时间长,低谷期慢SQL,周边业务慢SQL