合理利用锁机制优化MySQL

1 、 缩短锁定时间
缩短锁定时间,短短几个字,说起来确实听容易的,但实际做起来恐怕就并不那么简单了。如何让锁定时间尽可能的短呢?唯一的办法就是让我们的 Query 执行时间尽可能的短。
a) 尽两减少大的复杂 Query ,将复杂 Query 分拆成几个小的 Query 分布进行;
b) 尽可能的建立足够高效的索引,让数据检索更迅速;
c) 尽量让 MyISAM 存储引擎的表只存放必要的信息,控制字段类型;

d) 利用合适的机会优化 MyISAM 表数据文件;

2 、 分离能并行的操作
说到 MyISAM 的表锁,而且是读写互相阻塞的表锁,可能有些人会认为在 MyISAM 存储引擎的表上就 只能是完全的串行化,没办法再并行了。大家不要忘记了, MyISAM 的存储引擎还有一个非常有用的特性,那就是 Concurrent Insert (并发插入)的特性。MyISAM 存储引擎有一个控制是否打开 Concurrent Insert 功能的参数选项: concurrent_insert , 可
以设置为 0 , 1 或者 2 。三个值的具体说明如下:
a) concurrent_insert=2 ,无论 MyISAM 存储引擎的表数据文件的中间部分是否存在因为删除数据
而留下的空闲空间,都允许在数据文件尾部进行 Concurrent Insert;
b) concurrent_insert=1 ,当 MyISAM 存储引擎表数据文件中间不存在空闲空间的时候,可以从文
件尾部进行 Concurrent Insert;
c) concurrent_insert=0 ,无论 MyISAM 存储引擎的表数据文件的中间部分是否存在因为删除数据
而留下的空闲空间,都不允许 Concurrent Insert 。

3 、合理利用读写优先级
在本章各种锁定分析一节中我们了解到了 MySQL 的表级锁定对于读和写是有不同优先级设定的,默认情况下是写优先级要大于读优先级。所以,如果我们可以根据各自系统环境的差异决定读与写的优先级。如果我们的系统是一个以读为主,而且要优先保证查询性能的话,我们可以通过设置系统参数选项low_priority_updates=1 ,将写的优先级设置为比读的优先级低,即可让告诉 MySQL 尽量先处理读请求。当然,如果我们的系统需要有限保证数据写入的性能的话,则可以不用设置 low_priority_updates参数了。

Innodb 行锁优化建议:
Innodb 存储引擎由于实现了行级锁定,虽然在锁定机制的实现方面所带来的性能损耗可能比表级锁定会要更高一些,但是在整体并发处理能力方面要远远优于 MyISAM 的表级锁定的。当系统并发量较高的时候, Innodb 的整体性能和 MyISAM 相比就会有比较明显的优势了。但是, Innodb 的行级锁定同样也有 其脆弱的一面,当我们使用不当的时候,可能会让 Innodb 的整体性能表现不仅不能比 MyISAM 高,甚至可 能会更差。

合理利用 Innodb 的行级锁定,做到扬长避短,我们必须做好以下工作:

a) 尽可能让所有的数据检索都通过索引来完成,从而避免 Innodb 因为无法通过索引键加锁而升级为表级锁定;

b) 合理设计索引,让 Innodb 在索引键上面加锁的时候尽可能准确,尽可能的缩小锁定范围,避免造成不必要的锁定而影响其他 Query 的执行;
c) 尽可能减少基于范围的数据检索过滤条件,避免因为间隙锁带来的负面影响而锁定了不该锁定的记录;
d) 尽量控制事务的大小,减少锁定的资源量和锁定时间长度;
e) 在业务环境允许的情况下,尽量使用较低级别的事务隔离,以减少 MySQL 因为实现事务隔离级别所带来的附加成本;

由于 Innodb 的行级锁定和事务性,所以肯定会产生死锁,下面是一些比较常用的减少死锁产生概率的小建议,读者朋友可以根据各自的业务特点针对性的尝试:
a) 类似业务模块中,尽可能按照相同的访问顺序来访问,防止产生死锁;
b) 在同一个事务中,尽可能做到一次锁定所需要的所有资源,减少死锁产生概率;
c) 对于非常容易产生死锁的业务部分,可以尝试使用升级锁定颗粒度,通过表级锁定来减少死锁产生的概率;

MySQL 实现的表级锁定的争用状态变量:
mysql> show status like 'table%';

这里有两个状态变量记录 MySQL 内部表级锁定的情况,两个变量说明如下:
● Table_locks_immediate :产生表级锁定的次数;
● Table_locks_waited :出现表级锁定争用而发生等待的次数;
两个状态值都是从系统启动后开始记录,没出现一次对应的事件则数量加 1 。如果这里的Table_locks_waited 状态值比较高,那么说明系统中表级锁定争用现象比较严重,就需要进一步分析为什么会有较多的锁定资源争用了。

对于 Innodb 所使用的行级锁定,系统中是通过另外一组更为详细的状态变量来记录的,如下:
mysql> show status like 'innodb_row_lock%';

Innodb 的行级锁定状态变量不仅记录了锁定等待次数,还记录了锁定总时长,每次平均时长,以及最大时长,此外还有一个非累积状态量显示了当前正在等待锁定的等待数量。对各个状态量的说明如下:
● Innodb_row_lock_current_waits :当前正在等待锁定的数量;
● Innodb_row_lock_time :从系统启动到现在锁定总时间长度;
● Innodb_row_lock_time_avg :每次等待所花平均时间;
● Innodb_row_lock_time_max :从系统启动到现在等待最常的一次所花的时间;
● Innodb_row_lock_waits :系统启动后到现在总共等待的次数;
对于这 5 个状态变量,比较重要的主要是 Innodb_row_lock_time_avg (等待平均时长),Innodb_row_lock_waits (等待总次数)以及 Innodb_row_lock_time (等待总时长)这三项。尤其是当 等待次数很高,而且每次等待时长也不小的时候,我们就需要分析系统中为什么会有如此多的等待,然后根据分析结果着手指定优化计划。

此外, Innodb 出了提供这五个系统状态变量之外,还提供的其他更为丰富的即时状态信息供我们分析使用。可以通过如下方法查看:
1. 通过创建 Innodb Monitor 表来打开 Innodb 的 monitor 功能:
mysql> create table innodb_monitor(a int) engine=innodb;

2. 然后通过使用 “ SHOW INNODB STATUS ” 查看细节信息(由于输出内容太多就不在此记录了);可能会有读者朋友问为什么要先创建一个叫 innodb_monitor 的表呢?因为创建该表实际上就是告诉Innodb 我们开始要监控他的细节状态了,然后 Innodb 就会将比较详细的事务以及锁定信息记录进入MySQL 的 error log 中,以便我们后面做进一步分析使用。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值