Skip to content

Commit f39f8fb

Browse files
committed
更新MySQL锁类型
1 parent 976e833 commit f39f8fb

1 file changed

Lines changed: 283 additions & 4 deletions

File tree

notes/MySQL.md

Lines changed: 283 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -69,8 +69,8 @@
6969
- [6. 什么情况下使用索引](#6-什么情况下使用索引)
7070
- [10. 主键、外键和索引的区别](#10-主键外键和索引的区别)
7171
- [11. 聚集索引与非聚集索引](#11-聚集索引与非聚集索引)
72-
- [12. 数据库中的分页查询语句怎么写?【阿里面经】](#12-数据库中的分页查询语句怎么写阿里面经)
73-
- [13. 常用的数据库有哪些?Redis用过吗?【阿里面经】](#13-常用的数据库有哪些redis用过吗阿里面经)
72+
- [12. 数据库中的分页查询语句怎么写,如何优化](#12-数据库中的分页查询语句怎么写如何优化)
73+
- [13. 常用的数据库有哪些?Redis用过吗?](#13-常用的数据库有哪些redis用过吗)
7474
- [14. Redis的数据结构](#14-redis的数据结构)
7575
- [15. 分库分表](#15-分库分表)
7676
- [1. 垂直切分](#1-垂直切分)
@@ -96,6 +96,14 @@
9696
- [3. 重构查询方式](#3-重构查询方式)
9797
- [1. 切分大查询](#1-切分大查询)
9898
- [2. 分解大连接查询](#2-分解大连接查询)
99+
- [18. 锁类型](#18-锁类型)
100+
- [1. 乐观锁](#1-乐观锁)
101+
- [2. 悲观锁](#2-悲观锁)
102+
- [3. 共享锁](#3-共享锁)
103+
- [4. 排它锁](#4-排它锁)
104+
- [5. 行锁](#5-行锁)
105+
- [6. 表锁](#6-表锁)
106+
- [7. 死锁](#7-死锁)
99107
- [第二部分:高性能MySQL实践](#第二部分高性能mysql实践)
100108
- [1. 如何解决秒杀的性能问题和超卖的讨论](#1-如何解决秒杀的性能问题和超卖的讨论)
101109
- [解决方案1](#解决方案1)
@@ -1075,7 +1083,7 @@ https://www.cnblogs.com/s-b-b/p/8334593.html
10751083

10761084

10771085

1078-
## 12. 数据库中的分页查询语句怎么写?【阿里面经】
1086+
## 12. 数据库中的分页查询语句怎么写,如何优化
10791087

10801088
- Mysql 的 limit 用法
10811089
- SELECT * FROM table LIMIT [offset,] rows | rows OFFSET offset
@@ -1084,7 +1092,7 @@ https://www.cnblogs.com/s-b-b/p/8334593.html
10841092

10851093

10861094

1087-
## 13. 常用的数据库有哪些?Redis用过吗?【阿里面经】
1095+
## 13. 常用的数据库有哪些?Redis用过吗?
10881096

10891097
- 常用的数据库有哪些?Redis用过吗?
10901098
- 常用的数据库
@@ -1359,8 +1367,279 @@ SELECT * FROM post WHERE post.id IN (123,456,567,9098,8904);
13591367

13601368

13611369

1370+
## 18. 锁类型
13621371

1372+
MySQL/InnoDB 的加锁,一直是一个面试中常问的话题。例如,数据库如果有高并发请求,如何保证数据完整性?产生死锁问题如何排查并解决?在工作过程中,也会经常用到,乐观锁,排它锁等。
13631373

1374+
注:MySQL 是一个支持插件式存储引擎的数据库系统。下面的所有介绍,都是基于 InnoDB 存储引擎,其他引擎的表现,会有较大的区别。
1375+
1376+
**版本查看**
1377+
1378+
```mysql
1379+
select version();
1380+
```
1381+
1382+
**存储引擎查看**
1383+
1384+
MySQL 给开发者提供了查询存储引擎的功能,我这里使用的是 MySQL5.6.4,可以使用:
1385+
1386+
```mysql
1387+
SHOW ENGINES
1388+
```
1389+
1390+
1391+
1392+
### 1. 乐观锁
1393+
1394+
用数据版本(Version)记录机制实现,这是乐观锁最常用的一种实现方式。何谓数据版本?即为数据增加一个版本标识,一般是通过为数据库表增加一个数字类型的 “version” 字段来实现。当读取数据时,将version字段的值一同读出,数据每更新一次,对此version值加1。当我们提交更新的时候,判断数据库表对应记录的当前版本信息与第一次取出来的version值进行比对,如果数据库表当前版本号与第一次取出来的version值相等,则予以更新,否则认为是过期数据。
1395+
1396+
**举例**
1397+
1398+
1、数据库表设计
1399+
1400+
三个字段,分别是 id,value,version
1401+
1402+
```mysql
1403+
select id,value,version from TABLE where id=#{id}
1404+
```
1405+
1406+
2、每次更新表中的value字段时,为了防止发生冲突,需要这样操作
1407+
1408+
```mysql
1409+
update TABLE
1410+
set value=2,version=version+1
1411+
where id=#{id} and version=#{version};
1412+
```
1413+
1414+
1415+
1416+
### 2. 悲观锁
1417+
1418+
与乐观锁相对应的就是悲观锁了。悲观锁就是在操作数据时,认为此操作会出现数据冲突,所以在进行每次操作时都要通过获取锁才能进行对相同数据的操作,这点跟 Java 中的 synchronized 很相似,所以悲观锁需要耗费较多的时间。另外与乐观锁相对应的,悲观锁是由数据库自己实现了的,要用的时候,我们直接调用数据库的相关语句就可以了。
1419+
1420+
说到这里,由悲观锁涉及到的另外两个锁概念就出来了,它们就是**共享锁****排它锁****共享锁和排它锁是悲观锁的不同的实现**,它俩都属于悲观锁的范畴。
1421+
1422+
1423+
1424+
以排它锁为例:
1425+
1426+
要使用悲观锁,我们必须关闭 mysql 数据库的自动提交属性,因为 MySQL 默认使用 autocommit 模式,也就是说,当你执行一个更新操作后,MySQL 会立刻将结果进行提交。
1427+
1428+
我们可以使用命令设置 MySQL 为非 autocommit 模式:
1429+
1430+
```mysql
1431+
set autocommit=0;
1432+
# 设置完autocommit后,我们就可以执行我们的正常业务了。具体如下:
1433+
1434+
# 1. 开始事务 (三者选一就可以)
1435+
begin; / begin work; / start transaction;
1436+
1437+
# 2. 查询表信息
1438+
select status from TABLE where id=1 for update;
1439+
1440+
# 3. 插入一条数据
1441+
insert into TABLE (id,value) values (2,2);
1442+
1443+
# 4. 修改数据为
1444+
update TABLE set value=2 where id=1;
1445+
1446+
# 5. 提交事务
1447+
commit;/commit work;
1448+
```
1449+
1450+
1451+
1452+
### 3. 共享锁
1453+
1454+
共享锁又称**读锁**(read lock),是读取操作创建的锁。其他用户可以并发读取数据,但任何事务都不能对数据进行修改(获取数据上的排他锁),直到已释放所有共享锁。
1455+
1456+
如果事务 T 对数据 A 加上共享锁后,则其他事务只能对 A 再加共享锁,不能加排他锁。获得共享锁的事务只能读数据,不能修改数据
1457+
1458+
打开第一个查询窗口
1459+
1460+
```mysql
1461+
#三者选一就可以
1462+
begin; / begin work; / start transaction;
1463+
1464+
SELECT * from TABLE where id = 1 lock in share mode;
1465+
```
1466+
1467+
然后在另一个查询窗口中,对 id 为 1 的数据进行更新
1468+
1469+
```mysql
1470+
update TABLE set name="www.souyunku.com" where id =1;
1471+
```
1472+
1473+
此时,操作界面进入了卡顿状态,过了超时间,提示错误信息
1474+
1475+
如果在超时前,执行 `commit`,此更新语句就会成功。
1476+
1477+
```mysql
1478+
[SQL]update test_one set name="www.souyunku.com" where id =1;
1479+
[Err] 1205 - Lock wait timeout exceeded; try restarting transaction
1480+
```
1481+
1482+
加上共享锁后,也提示错误信息
1483+
1484+
```mysql
1485+
update test_one set name="www.souyunku.com" where id =1 lock in share mode;
1486+
[SQL]update test_one set name="www.souyunku.com" where id =1 lock in share mode;
1487+
[Err] 1064 - You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'lock in share mode' at line 1
1488+
```
1489+
1490+
在查询语句后面增加 `lock in share mode`,MySQL 会对查询结果中的每行都加共享锁,当没有其他线程对查询结果集中的任何一行使用排他锁时,可以成功申请共享锁,否则会被阻塞。其他线程也可以读取使用了共享锁的表,而且这些线程读取的是同一个版本的数据。
1491+
1492+
加上共享锁后,对于 `update,insert,delete` 语句会自动加排它锁。
1493+
1494+
### 4. 排它锁
1495+
1496+
排他锁 exclusive lock(也叫 writer lock)又称**写锁**
1497+
1498+
**排它锁是悲观锁的一种实现,在上面悲观锁也介绍过**
1499+
1500+
若事务 1 对数据对象 A 加上 X 锁,事务 1 可以读 A 也可以修改 A,其他事务不能再对 A 加任何锁,直到事物 1 释放 A 上的锁。这保证了其他事务在事物 1 释放 A 上的锁之前不能再读取和修改 A。排它锁会阻塞所有的排它锁和共享锁
1501+
1502+
读取为什么要加读锁呢:防止数据在被读取的时候被别的线程加上写锁
1503+
1504+
使用方式:在需要执行的语句后面加上 `for update` 就可以了
1505+
1506+
1507+
1508+
### 5. 行锁
1509+
1510+
行锁又分**共享锁****排他锁**,由字面意思理解,就是给某一行加上锁,也就是一条记录加上锁。
1511+
1512+
**注意**:行级锁都是基于索引的,如果一条SQL语句用不到索引是不会使用行级锁的,会使用表级锁。
1513+
1514+
**共享锁:**
1515+
1516+
名词解释:共享锁又叫做读锁,所有的事务只能对其进行读操作不能写操作,加上共享锁后在事务结束之前其他事务只能再加共享锁,除此之外其他任何类型的锁都不能再加了。
1517+
1518+
```mysql
1519+
#结果集的数据都会加共享锁
1520+
SELECT * from TABLE where id = "1" lock in share mode;
1521+
```
1522+
1523+
**排他锁:**
1524+
1525+
名词解释:若某个事物对某一行加上了排他锁,只能这个事务对其进行读写,在此事务结束之前,其他事务不能对其进行加任何锁,其他进程可以读取,不能进行写操作,需等待其释放。
1526+
1527+
```mysql
1528+
select status from TABLE where id=1 for update;
1529+
```
1530+
1531+
可以参考之前演示的共享锁,排它锁语句
1532+
1533+
由于对于表中 id 字段为主键,就也相当于索引。执行加锁时,会将 id 这个索引为 1 的记录加上锁,那么这个锁就是行锁。
1534+
1535+
1536+
1537+
### 6. 表锁
1538+
1539+
如何加表锁
1540+
1541+
innodb 的行锁是在有索引的情况下,没有索引的表是锁定全表的.
1542+
1543+
**Innodb中的行锁与表锁**
1544+
1545+
前面提到过,在 Innodb 引擎中既支持行锁也支持表锁,那么什么时候会锁住整张表,什么时候或只锁住一行呢?
1546+
只有通过索引条件检索数据,InnoDB 才使用行级锁,否则,InnoDB 将使用表锁!
1547+
1548+
在实际应用中,要特别注意 InnoDB 行锁的这一特性,不然的话,可能导致大量的锁冲突,从而影响并发性能。
1549+
1550+
行级锁都是基于索引的,如果一条 SQL 语句用不到索引是不会使用行级锁的,会使用表级锁。行级锁的缺点是:由于需要请求大量的锁资源,所以速度慢,内存消耗大。
1551+
1552+
1553+
1554+
### 7. 死锁
1555+
1556+
死锁(Deadlock)
1557+
所谓死锁:是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。此时称系统处于死锁状态或系统产生了死锁,这些永远在互相等待的进程称为死锁进程。由于资源占用是互斥的,当某个进程提出申请资源后,使得有关进程在无外力协助下,永远分配不到必需的资源而无法继续运行,这就产生了一种特殊现象死锁。
1558+
1559+
解除正在死锁的状态有两种方法:
1560+
1561+
**第一种**
1562+
1563+
1. 查询是否锁表
1564+
1565+
```mysql
1566+
show OPEN TABLES where In_use > 0;
1567+
```
1568+
1569+
2. 查询进程(如果您有SUPER权限,您可以看到所有线程。否则,您只能看到您自己的线程)
1570+
1571+
```mysql
1572+
show processlist
1573+
```
1574+
1575+
3. 杀死进程id(就是上面命令的id列)
1576+
1577+
```mysql
1578+
kill id
1579+
```
1580+
1581+
**第二种**
1582+
1583+
1. 查看当前的事务
1584+
1585+
```mysql
1586+
SELECT * FROM INFORMATION_SCHEMA.INNODB_TRX;
1587+
```
1588+
1589+
2. 查看当前锁定的事务
1590+
1591+
```mysql
1592+
SELECT * FROM INFORMATION_SCHEMA.INNODB_LOCKS;
1593+
```
1594+
1595+
3. 查看当前等锁的事务
1596+
1597+
```mysql
1598+
SELECT * FROM INFORMATION_SCHEMA.INNODB_LOCK_WAITS;
1599+
```
1600+
1601+
**杀死进程**
1602+
1603+
```mysql
1604+
kill 进程ID
1605+
```
1606+
1607+
如果系统资源充足,进程的资源请求都能够得到满足,死锁出现的可能性就很低,否则就会因争夺有限的资源而陷入死锁。其次,进程运行推进顺序与速度不同,也可能产生死锁。
1608+
产生死锁的四个必要条件:
1609+
1610+
1. 互斥条件:一个资源每次只能被一个进程使用。
1611+
1612+
2. 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
1613+
1614+
3. 不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
1615+
1616+
4. 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。
1617+
1618+
虽然不能完全避免死锁,但可以使死锁的数量减至最少。将死锁减至最少可以增加事务的吞吐量并减少系统开销,因为只有很少的事务回滚,而回滚会取消事务执行的所有工作。由于死锁时回滚而由应用程序重新提交。
1619+
1620+
**下列方法有助于最大限度地降低死锁:**
1621+
1622+
1. 按同一顺序访问对象
1623+
1624+
2. 避免事务中的用户交互
1625+
1626+
3. 保持事务简短并在一个批处理中
1627+
1628+
4. 使用低隔离级别
1629+
1630+
5. 使用绑定连接
1631+
1632+
1633+
1634+
说明:间隙锁相关锁知识待补充
1635+
1636+
1637+
1638+
参考资料:
1639+
1640+
- [Mysql锁机制简单了解一下 - Java面试通关手册 - SegmentFault 思否](https://segmentfault.com/a/1190000015219003#articleHeader0)
1641+
1642+
- [锁概念的理解 - 搜云库 - SegmentFault 思否](https://segmentfault.com/a/1190000015815061#articleHeader7)
13641643

13651644

13661645

0 commit comments

Comments
 (0)