首页 > 数据库 >MySQL锁:InnoDB行锁需要避免的坑

MySQL锁:InnoDB行锁需要避免的坑

时间:2022-11-28 11:13:29浏览次数:64  
标签:加锁 行锁 索引 InnoDB MySQL id

前言

  换了工作之后,接近半年没有发博客了(一直加班),emmmm.....今天好不容易有时间,记录下工作中遇到的一些问题,接下来应该重拾知识点了。因为新公司工作中MySQL库经常出现查询慢,锁等待,节点挂掉........等一系列问题。导致每个程序员头都很大,一味抱怨“为什么我就查一条数据这么卡”,"我TM加了索引的啊,怎么还怎么慢"...........我想默默说的是,大部分MySQL出现锁等待,查询奇慢的情况基本都是因为SQL写的不好(有坑),或者数据表设计的不完善。对,不用想!这些所有的坑很大一部分都是自己造成的。那么是什么原因造成的,大部分只是抱怨,而不去关注MySQL的一些细节问题,比如:MySQL行锁的细节,什么情况下会使用表锁等。所以今天先讨论记录下InnoDB特有的行锁的一些细节,加强认识。

  InnoDB不同于MyISAM最大的两个特点就是:一是支持事务,二是支持行锁;毋庸置疑,因为这两个特性大部分都采用InnoDB引擎,其中的支持行锁就是InnoDB适合多并发优势所在,但是行锁的一些细节没有深入理解过的话,可能会造成一定的误解,造成“看似命中索引,走行锁,结果却是表锁,最终导致锁等待情况”。

 


一、InnoDB行锁的实现方式

  通过给索引上的索引项加锁来实现的,也就意味着:只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁。这一点在实际应用中特别需要注意,不然的话可能导致大量的锁冲突,从而影响引发并发性能

  实验一:对没有索引的加锁,导致表锁

   1)准备工作:建tab_no_index表,表中无任何索引,并插入数据

    

 

   2)Session_1: 我们给id=1的行加上排它锁(for update),由于id没有索引,实际上是表级锁;

    

   3)Session_2:我们给id=2的行加上排它锁(for update),由于id没有索引,所以去申请表级锁,但是却出现了锁等待!原因就是在没有索引的情况下,InnoDB只能使用表锁。

    

    备注:MySQL中的for update 仅适用于InnoDB(因为是只有此引擎才有行级锁),并且必须开启事务,在begin与commit之间才生效。for update是在数据库中上锁用的,可以为数据库中的行上一个排它锁。当一个事务的操作未完成时候,其他事务可以对这行读取但是不能写入或更新,只能等该事务Rollback, Commit, Lost connection…

   实验二:对有索引的键值加锁,会对所有涉及到的数据行加锁

    1)准备工作:对id建索引如下

     

    2)Session_1:此时id是有索引的,我们对id=1 and name=1的一行加排它锁;

    

    3)Session_2:访问不同于Session_1的id=1, name=5行,但是索引键值是一样的,照样等待锁,锁冲突了。

    

   实验三:多个索引时,不同的事务可以使用不同的索引锁定不同的行,不论什么索引,InnoDB都会使用行锁对数据加锁(对有索引的行数据)

    1)准备工作:对tab_no_index追加name索引:alter table tab_no_index add index name(name);

    

    2)Session_1:开启事务对id=1的行加排它锁,即对name=1与name=5两个数据加锁。

    

    3)Session_2:开启事务对name=2行加锁,因为该数据没有被加锁,索引可以获得锁

    

    4)Session_3:再对name=5的数据进行加锁,由于该数据记录已被Session_1锁定,所以等待获得锁。

   

  注意事项:即便使用了索引,但还是要看MySQL具体对SQL的执行计划,不一定能使用到

    如我们对实验三对name='2'进行加锁,误以为name是int类型,本来name是有索引的,但是最后结果导致表锁:

    

  

  具体请看MySQL的索引情况。具体可以参考之前我的一篇博文MySQL优化(1)--------常用的优化步骤MySQL优化(2)--------常用优化

 

二、间隙锁(Next-Key锁)

  当用范围条件而不是相等条件检索数据,并请求共享或者排它锁的时候,InnoDB会给符合条件的已有数据记录的索引项加锁;对于不在范围内的但并不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个间隙加锁,这就是所谓的间隙锁

  如:select * from where id>100 for update 对id大于100的数据对加锁,但是此时数据中id只有1,2….100,101,不仅对存在的101的记录加锁,还会对大于101不存在的数据的间隙加锁。

   此外,对使用相等条件请求给一个不存在的记录加锁,InnoDB也会使用间隙锁,如下:

  Session_1:对不存在的id=6的记录加锁

  

 

  Session_2:插入id=6的记录,也会出现锁等待

  

 

三、什么时候使用表锁?

  对于InnoDB表,在绝大部分情况下都应该使用行锁,因为事务和行锁往往是我们之所以选择InnoDB表的理由,但在个别情况下也使用表级锁;

  1)事务需要更新大部分或全部数据,表又比较大,如果使用默认的行锁,不仅这个事务执行效率低,而且可能造成其他事务长时间等待和锁冲突;

  2)事务涉及多个表,比较复杂,很可能引起死锁,造成大量事务回滚。

使用表锁需要注意几点:

  1)使用LOCK TABLES虽然可以给InnoDB加表级锁,表级锁不是InnoDB存储引擎层管理的,而是由其上一层MySQL Server负责的

  2)在用LOCK TABLES对InnoDB表加锁时需要注意,要将Autocommit设置为0,否则MySQL不会给表加锁;事务结束前,不要用UNLOCK TABLES释放表锁,因为UNLOCK_TABLES隐含提交事务;COMMIT或ROLLBACK并不能释放用LOCK TABLES加表级锁。

   SET AUTOCOMMIT=0;

   LOCK TABLES table1 WRITE, table2 READ,...;

   [do something....]

   COMMIT;

   UNLOCK TABLES;

 

 


 

总结:

  • 从设计之初,就应该建立良好的索引机制,避免对关键字段搜索时造成表锁;

  • 避免长时间事务未提交等情况,导致锁冲突,死锁等情况;

  • 不要老是抱怨数据库有问题,应该从自身写的SQL分析出发,学会分析(数据库不行大部分是因为SQL写的有问题,没错,是自身问题);

  • 不要总是觉得这是DBA该做的事,开发者应该学会基本的SQL常识(如MySQL的最左索引,回表,索引覆盖等知识),学会基本的优化步骤。

主要参考资料:《深入浅出MySQL》(有需要PDF电子书的伙伴可以评论或者私信我)

https://www.cnblogs.com/jian0110/p/12721924.html 

标签:加锁,行锁,索引,InnoDB,MySQL,id
From: https://www.cnblogs.com/wxdlut/p/16931652.html

相关文章

  • mysql删除所有表中数据
    建立存储过程dropprocedureifexistsdel_all_tb;delimiter$$createproceduredel_all_tb(dbchar(20))begindeclaredoneintdefault0;declaretbc......
  • mysql只更改字段年月日时间,时分秒不变
    数据库导入数据,年月日出现问题了,如图所示,只能通过sql更改年月日。想修改一列时间,将其年月日修改为自己想要的时间执行如下sql:UPDATEyieldsetdataTime=ADDTIME(DAT......
  • Mysql 直接拷贝数据库文件导致表不显示的问题
    前言:最近有一个需求,需要迁移数据库中的其中一个库,需要迁移的那个数据库占用了700多G的空间,所以采用直接拷贝数据库文件的方式,拷贝到另一台服务器后发现表不显示,记录本次问......
  • mysql 数据库管理
    一、常用的数据类型二、数据库管理2.1创建数据库和表-create2.2删除数据库和表-drop  一、常用的数据类型类型解释举例int整型......
  • MySQL对时间戳的格式化
    时间转时间戳selectunix_timestamp('2022-02-2222:22:22');时间戳转时间selectfrom_unixtime(1645539742);格式化SELECTfrom_unixtime(1645539742,'%Y-%m-%d%H......
  • 【mysql】mysql---单行函数、多行函数
    单行函数:1.字符串函数2.数值函数3.日期时间函数4.流程函数5.JSON函数6.其他函数4.流程函数--if(condition,t,f)双分支结构selectempno,sal,if(sal>5000,"高薪",......
  • 【mysql】一些知识的补充
    mysql8.01.创建数据库表##这是一个单行注释/*多行注释多行注释多行注释*//*建立一张用来存储学生信息的表字段包含学号、姓名、性别,年龄、入学日期、班级,email......
  • 在mysql bench 客户端链接mysql时 出现拒绝访问
    在mysqlbench客户端链接mysql时出现拒绝访问-----经过排查是因为新安装的系统centosstream9的防火墙是开启的,且默认端口3306是关闭的,需要对该端口号进行开放。就......
  • 第15节-MySQL用户权限
    1、用户管理1.1、查询用户usemysql--5.6以下的版本selecthost,user,passwordfromuser;--5.7以上的版本selecthost,user,authentication_stringfromuser;......
  • mysql添加远程连接功能
    有些情况,我们需要让测试服务器的本地数据库能被外部访问,方便本地数据库可视化工具对其数据的管理。但是,这也带来了一些风险,若被黑客知道了你的服务器IP和数据库账号密码,则......