mysql锁机制总结,以及优化建议

一、锁概述和分类

 

 

二、表锁

偏向MyISAM存储引擎,开销小,加锁快;无死锁;锁定粒度大,发生锁冲突的概率最高,并发度最低。

【手动增加表锁】

lock table 表名字1 read(write),表名字2 read(write),其它;

【查看表上加过的锁】

show open tables;

【释放表锁】

unlock tables;

演示:

mysql> select * from mylock;
+----+------+
| id | name |
+----+------+
|  1 | a    |
|  2 | b    |
|  3 | c    |
|  4 | d    |
|  5 | e    |
+----+------+
5 rows in set (0.00 sec)

# 给mylock表加读锁,给t1表加写锁
mysql> lock table mylock read, t1 write;
Query OK, 0 rows affected (0.02 sec)

# 查看已经加锁的表, 下面的结果省略了很多行
mysql> show open tables;
+--------------------+------------------------------------------------------+--------+-------------+
| Database           | Table                                                | In_use | Name_locked |
+--------------------+------------------------------------------------------+--------+-------------+
| mysqlad            | t1                                                   |      1 |           0 |
| performance_schema | events_transactions_current                          |      0 |           0 |
| performance_schema | events_statements_summary_by_program                 |      0 |           0 |
| performance_schema | events_waits_summary_by_host_by_event_name           |      0 |           0 |
| mysqlad            | mylock                                               |      1 |           0 |
| performance_schema | file_sum
121 rows in set (0.00 sec)

# 释放表锁
mysql> unlock tables;
Query OK, 0 rows affected (0.00 sec)

读锁案例:下面通过两个会话窗口来演示对mylock表加读锁之后的效果:

演示对mylock加写锁:

通过上面的实验,可以发现:

MyISAM在执行查询语句(SELECT)前,会自动给涉及的所有表加读锁,在执行增删改操作前,会自动给涉及的表加写锁。

MySQL的表级锁有两种模式:

结合上表,所以对MyISAM表进行操作,会有以下情况:

  • 1、对MyISAM表的读操作(加读锁),不会阻塞其他进程对同一表的读请求,但会阻塞对同一表的写请求。只有当读锁释放后,才会执行其它进程的写操作。
  • 2、对MyISAM表的写操作(加写锁),会阻塞其他进程对同一表的读和写操作,只有当写锁释放后,才会执行其它进程的读写操作。

简而言之,就是读锁会阻塞写,但是不会堵塞读。而写锁则会把读和写都堵塞。

总结:

  • 可以通过show open tables来查看哪些表被枷锁了;
  • 如何分析表锁定,可以通过检查table_locks_waitedtable_locks_immediate状态变量来分析系统上的表锁定;

 

 

 

这里有两个状态变手记录MySQL内部表级锁定的情况,两个变量说明如下:

Table_locks_immediate: 产生表级锁定的次数,表示可以立即获取锁的查询次数,每立即获取锁值加1 ;

Table_locks_waited: 出现表级锁定争用而发生等待的次数(不能立即获取锁的次数,每等待一次锁值加1),此值高则说明存在着较严重的表级锁争用情况;

总结: MyISAM的读写锁调度是优先,这也是MyISAM不适合做写为主表的引擎。 因为写锁后,其他线程不能做任何操作,大量的更新会使查询很难得到锁,从而造成永远阻塞。

三、行锁

特点:

  • 偏向InnoDB存储引擎,开销大,加锁慢;
  • 会出现死锁;
  • 锁定粒度最小,发生锁冲突的概率最低,并发度也最高。
InnoDB与MyISAM的最大不同有两点: 一是支持事务(TRANSACTION);二是采用了行级锁
Innodb存储引擎由于实现了行级锁定,虽然在锁定机制的实现方面所带来的性能损耗可能比表级锁定会要更高一些,但是在整体并发处理能力方面要远远优于MyISAM的表级锁定的。当系统并发量较高的时候,Innodb的整体性能和MyISAM相比就会有比较明显的优势了
但是,Innodb的行级锁定同样也有其脆弱的一面,当我们使用不当的时候,可能会让Innodb的整体性能表现不仅不能比MyISAM高,甚至可能会更差。

演示案例,建表SQL:

create table test_innodb_lock (a int(11),b varchar(16))engine=innodb;

insert into test_innodb_lock values(1,'b2');
insert into test_innodb_lock values(3,'3');
insert into test_innodb_lock values(4,'4000');
insert into test_innodb_lock values(5,'5000');
insert into test_innodb_lock values(6,'6000');
insert into test_innodb_lock values(7,'7000');
insert into test_innodb_lock values(8,'8000');
insert into test_innodb_lock values(9,'9000');
insert into test_innodb_lock values(1,'b1');

# 创建两个索引
create index test_innodb_a_ind on test_innodb_lock(a);
create index test_innodb_lock_b_ind on test_innodb_lock(b);

# 查询结果
mysql> select * from test_innodb_lock;
+------+------+
| a    | b    |
+------+------+
|    1 | b2   |
|    3 | 3    |
|    4 | 4000 |
|    5 | 5000 |
|    6 | 6000 |
|    7 | 7000 |
|    8 | 8000 |
|    9 | 9000 |
|    1 | b1   |
+------+------+
9 rows in set (0.00 sec)

测试: 读己之所写

 

 

然后看session_1session_2同时更新a = 4的情况:

更新但是不提交,即没有commit

session_2被阻塞,只能等待

提交更新

mysql> commit;解除阻塞

 

但是如果两个会话不是更新同一行呢?

如果不是更新同一行,则就算在session_1没有commit的时候,session_2也不会阻塞。

 

 

尽可能让所有数据检索都通过索引来完成,避免无索引行锁升级为表锁

举个例子: 因为我们的bvarchar类型的,更新的时候我故意将b的单引号去掉,此时MYSQL底层自动类型转换,但是此时就会导致索引失效,然后我们看下面,就会导致我们的行锁变成了表锁,从而导致阻塞等待。

 

间隙锁带来的插入问题:

 

【什么是间隙锁】

当我们用范围条件而不是相等条件检索数据,并请求共享或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但并不存在的记录,叫做“间隙(GAP)”,

InnoDB也会对这个“间隙”加锁(不放过一个),这种锁机制就是所谓的间隙锁(GAP Lock)。

【危害】

因为Query执行过程中通过过范围查找的话,他会锁定整个范围内所有的索引键值,即使这个键值并不存在。

间隙锁有一个比较致命的弱点,就是当锁定一个范围键值之后,即使某些不存在的键值也会被无辜的锁定,而造成在锁定的时候无法插入锁定键值范围内的任何数据。在某些场景下这可能会对性能造成很大的危害。

面试题:常考如何锁定一行。
使用 for update

 

【如何分析行锁定】

通过检查InnoDB_row_lock状态变量来分析系统上的行锁的争夺情况

mysql>show status like 'innodb_row_lock%';

 

 

 

对各个状态量的说明如下:

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(等待总时长)这三项。

尤其是当等待次数很高,而且每次等待时长也不小的时候,我们就需要分析系统中为什么会有如此多的等待,然后根据分析结果着手指定优化计划。

最后可以通过SELECT * FROM information_schema.INNODB_TRX\G;来查询正在被锁阻塞的sql语句。

四、优化建议

  • 尽可能让所有数据检索都通过索引来完成,避免无索引行锁升级为表锁
  • 尽可能较少检索条件,避免间隙锁
  • 尽量控制事务大小,减少锁定资源量和时间长度;
  • 锁住某行后,尽量不要去调别的行或表,赶紧处理被锁住的行然后释放掉锁;
  • 涉及相同表的事务,对于调用表的顺序尽量保持一致;
  • 在业务环境允许的情况下,尽可能低级别事务隔离;
免费Java高级资料需要自己领取,涵盖了Java、Redis、MongoDB、MySQL、Zookeeper、Spring Cloud、Dubbo高并发分布式等教程,一共30G。
传送门: https://mp.weixin.qq.com/s/JzddfH-7yNudmkjT0IRL8Q
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 从零开始带你成为MySQL实战优化高手(130节),是一套涵盖了MySQL数据库优化的全面学习教程。MySQL是一款常用的关系型数据库管理系统,优化MySQL的性能对于提高数据库操作的效率至关重要。 该教程的130节从基础知识开始,包括MySQL的安装和配置、基本SQL语句的使用,以及索引的原理和使用等。接着介绍了数据库表的设计优化和数据类型的选择,帮助读者了解如何优化数据库的结构。 在进一步深入学习中,该教程重点讲解了查询优化和索引优化的技巧。它详细介绍了MySQL查询执行的流程和性能影响因素,以及如何通过使用合适的索引和优化查询语句来提高查询效率。此外,还介绍了锁定机制和事务处理的优化方法,帮助读者更好地应对并发操作和高负载压力。 除了优化数据库性能外,该教程还涵盖了备份和恢复、监控和诊断等方面的内容。它介绍了MySQL的备份和恢复策略,以及如何通过监控工具来追踪和分析数据库的性能问题。此外,该教程还分享了一些常见的优化实践和技巧,帮助读者更好地应用所学知识。 通过学习该教程,读者将掌握MySQL数据库优化的核心概念和技能,并能够在实际工作中应用这些知识。无论是从事数据库管理、开发还是运维工作,通过成为MySQL实战优化高手,读者都能够提升工作效率,优化数据库性能,提供更好的用户体验。 ### 回答2: 要成为MySQL实战优化高手,需要通过学习和实践来提升自己的技能和经验。以下是从零开始带你成为MySQL实战优化高手的步骤和内容: 1. 基础知识学习:从MySQL的基本概念、架构和常用命令开始学习,理解数据库的基本原理和操作方法。 2. SQL语句优化:学习如何编写高效的SQL语句,包括合理选择表达式、索引的使用、避免全表扫描等技巧。 3. 索引设计:深入了解MySQL索引的原理和不同类型的索引使用场景,学习如何为表设计合适的索引,提高查询性能。 4. 查询性能优化:学习如何通过查询优化器、查询执行计划等工具来分析和优化查询语句,提高查询的执行效率。 5. 数据库参数调优:了解MySQL的常用参数设置和优化原则,学习如何根据实际需求和硬件配置来调整参数,提升数据库性能。 6. 数据库架构设计:学习如何合理划分数据库表、使用数据分区、设计合适的数据类型等,提高系统的扩展性和性能。 7. 事务和并发控制:了解事务的基本概念和特性,学习如何正确使用事务和锁机制,保证数据的一致性和并发性能。 8. 数据库备份与恢复:学习如何进行数据库的备份和恢复,包括全量备份、增量备份、物理备份和逻辑备份等方法。 9. 数据库监控和性能调优:学习如何使用MySQL自带的监控工具和第三方性能分析工具,对数据库进行实时监控和性能调优。 10. 实践项目:通过参与实际项目的开发和维护,积累实战经验,提升自己的实际操作能力和解决问题的能力。 总结来说,成为MySQL实战优化高手需要坚持学习和实践,不断提升自己的技能和经验。同时,多参与实际项目和团队协作,通过和他人的交流和互动,加速自己的成长和进步。 ### 回答3: 要成为MySQL实战优化高手,首先需要掌握MySQL的基本知识和使用方法。建议从学习数据库的基础知识开始,了解数据库的概念、关系型数据库和非关系型数据库等的特点和应用场景。 接下来,学习MySQL的安装和配置,掌握MySQL的常见命令行操作和GUI工具的使用。了解MySQL的体系结构和数据存储方式,熟悉MySQL的各种数据类型和索引类型。 学习MySQL的SQL语句,包括数据的增删改查操作、表的创建和修改、条件查询、多表查询、排序、分组等常用的SQL语句。熟悉MySQL的事务和锁机制,了解如何处理并发访问和脏读、幻读等问题。 进一步学习MySQL的性能优化技巧,包括索引设计、查询优化、SQL语句性能调优、数据库参数和配置优化等方面。学习如何通过explain查询计划、慢查询日志等工具进行性能分析和优化。 深入学习MySQL的高级特性,包括分区表、分布式数据库、存储过程、触发器、事件调度器等。了解MySQL的复制、备份和恢复机制,熟悉主从复制、多主复制等架构。 通过实战项目来提升自己的实战能力。可以尝试从零开始搭建一个具有一定规模和复杂度的MySQL数据库,模拟真实的应用场景,进行数据建模、查询优化、性能调优等工作。 不断阅读学习MySQL的相关技术文档、博客和社区,在实践中积累经验,与其他MySQL开发者和优化高手交流讨论问题,提升自己的专业水平。 最后,坚持学习和实践,持续关注MySQL的最新动态和发展趋势,保持学习的热情和好奇心,才能逐渐成为MySQL实战优化的高手。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值