MySQL - 无索引行锁升级为表锁

在这里插入图片描述


生猛干货

带你搞定MySQL实战,轻松对应海量业务处理及高并发需求,从容应对大场面试


无索引行锁升级为表锁演示

表结构

mysql> desc country;
+-------------+--------------+------+-----+---------+----------------+
| Field       | Type         | Null | Key | Default | Extra          |
+-------------+--------------+------+-----+---------+----------------+
| id          | int(11)      | NO   | PRI | NULL    | auto_increment |
| countryname | varchar(255) | YES  |     | NULL    |                |
| countrycode | varchar(255) | YES  |     | NULL    |                |
+-------------+--------------+------+-----+---------+----------------+
3 rows in set


索引信息

mysql> show index from country;
+---------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| Table   | Non_unique | Key_name | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment |
+---------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| country |          0 | PRIMARY  |            1 | id          | A         |           5 | NULL     | NULL   |      | BTREE      |         |               |
+---------+------------+----------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
1 row in set


这个表,只有主键索引,其他字段上未建立二级索引 。

现在使用没有建立索引的字段进行操作,观察其结果


操作演示

session1session2
begin 模拟开启事务
update country set countryname = ‘CCC’ where countrycode = ‘xxx’ ; --更新成功,但未提交事务
begin 模拟开启事务
update country set countryname = ‘DDD’ where countrycode = ‘anotherline’ ; ---- 一直被阻塞 ,直到超时 1205 - Lock wait timeout exceeded; try restarting transaction

我们知道锁主要是加在索引上,如果对非索引字段更新,行锁可能会变表锁 , 从上面的测试中也可以验证这个观点,第二个


结论

InnoDB的行锁是针对索引加的锁,不是针对记录加的锁 ,并且该索引不能失效,否则会从行锁升级为表锁 。

所以建表的时候 ,结合你的业务,如果有更新的操作,切记要对操作的字段建立索引,不然并发下这个问题就非常明显了


搞定MySQL

在这里插入图片描述

### 回答1: MySQL中的锁是指在执SQL语句时对某一数据进锁定,以防止其他会话对该修改或删除。当一个会话对某一更新或删除操作时,MySQL会为该加上级锁,直到事务提交或回滚后才会释放。 在什么情况下升级为表锁? 当多个会话同时对表中的不同更新或删除操作时,级锁可能会导致死锁问题,影响数据库性能。此时,MySQL会自动将锁升级为表锁,以避免死锁问题的发生。因此,当出现大量并发更新或删除操作时,MySQL可能会自动升级为表锁。 此外,也可以通过手动设置锁级别来控制MySQL的锁为。例如,可以将锁级别设置为READ COMMITTED(读取已提交),在该级别下,MySQL会对读取的数据共享锁,对更新和删除的数据排它锁。如果需要对整个表进操作,可以使用表级锁或表级共享锁。 ### 回答2: 在MySQL中,锁和表锁是两种不同级别的锁机制。通常情况下,MySQL会根据需要自动升级为表锁。主要有以下几种情况: 1. 锁定范围超出了级别:当事务需要锁定的数超过了MySQL设置的阈值时,MySQL会自动将锁升级为表锁。这样做是为了减少锁的数量,提高并发性能。 2. 不同事务涉及到的相互冲突:如果不同的事务需要锁定同一张表中的不同,并且它们之间有冲突,那么MySQL会将锁升级为表锁。这个冲突可能是由于事务之间的读写操作或写入冲突引起的。 3. 显式设置表锁:在一些特定情况下,我们可能需要显式地将锁升级为表锁。这可以通过使用LOCK TABLES语句来实现。但是,这个操作需要小心使用,因为表锁会限制其他事务的读写操作。 需要注意的是,MySQL锁和表锁是根据实际情况进升级的,并不是所有的锁都会自动升级为表锁。升级为表锁的条件是符合上述所述的情况。在实际应用中,我们需要根据具体的业务需求和场景,合理选择和使用锁和表锁,以提高数据库的并发性和性能。 ### 回答3: MySQL中的锁升级为表锁是在以下情况下发生的: 1. 事务涉及到的数量超过了innodb_locks_unsafe_for_binlog的阈值,默认值是30000。如果一个事务涉及到的数超过了这个阈值,锁就会自动升级为表锁。这是为了防止由于级锁过多导致事务无法进的情况。 2. 出现了锁等待超时。当一个事务等待锁的时间超过了设置的超时时间(innodb_lock_wait_timeout),锁也会自动升级为表锁。这个超时时间可以在MySQL的配置文件中进设置。 需要注意的是,锁升级为表锁可能会导致并发性能下降。因为表级锁会导致其他事务需要等待,无法并发执。因此,在设计数据库时,需要合理使用级锁和表级锁。对于并发读多写少的场景,可以使用锁来提高并发性能。对于并发写多的场景,可以考虑使用表锁来避免锁冲突。此外,可以使用合理的索引设计来减少锁竞争,提高并发性能。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小小工匠

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值