Mysql之数据库锁(表锁和行锁)详解
1. 什么是锁?
锁是计算机协调多个进程或线程并发访问某一资源的机制。
- 在数据库中,除传统的计算资源(如CPU、RAM、I/O等)的争用以外,数据也是一种共享资源,如何保证数据并发访问的一致性、有效性是所有数据库必须解决的一个问题。
- 锁冲突是影响数据库并发访问性能的一个重要因素,从这个角度来说,锁对数据库而言显得尤为重要,而且也更加复杂
生活举例:
2. 锁的分类
2.1 从对数据的操作类型分为:读锁(共享锁)和 写锁(排他锁)
读锁:针对同一份数据,对该数据的读操作可以同时进行且不受影响。
写锁:写操作未完成前,会阻断其他的读操作和写操作。
2.2 从对数据的操作粒度分为:表锁 和 行锁
表锁
表锁特点
- MylSAM引擎使用表锁,开销小,加锁快,无死锁,锁定力度大,发生锁冲突的概率最高。
- 并发度最低
- 不支持事务
3.建立数据
create table mylock (
id int not null primary key auto_increment,
name varchar(20) default ''
) engine myisam;insert into mylock(name) values('a');
insert into mylock(name) values('b');
insert into mylock(name) values('c');
insert into mylock(name) values('d');
insert into mylock(name) values('e');select * from mylock;
查看数据库锁
SHOW OPEN TABLES in hanyxx; --查看数据库hanyxx中的表是否加锁
LOCK TABLE book read , phone write -- book表加读锁,phone表加写锁
--全部解锁
UNLOCK TABLES;
3.1 手动加锁
释放锁
3.1加读锁
.
加表锁
读当前表,可以
改当前表,不可以
读其他表:不可以
表锁(读锁)
session_1给表加上表锁(读锁)
session_1和其他主机都可以读取该表的信息
session_1不能读取其他表的信息,但其他session可以读取库中其他表的信息
session_1不能对锁住的表进行修改
其他session对表进行修改,会被阻塞,直到表锁(读锁)被释放
此时session_2来对表修改,会被阻塞
总结:
读锁不会阻塞读,只会阻塞写。
写锁会阻塞读和写
3.2 写锁:排他锁
3.3行锁(重中之重)
行锁特点
- 开销大,枷锁慢,会出现死锁
- 锁定粒度最小,发生锁冲突概率最低,并发度最高
InnoDB和MyISAM最大的不同点
- 支持事务
- 采用行锁和外键
学习表锁之前先复习一下事务
1.事务具有ACID的属性
更新丢失
脏读
不可重复读
幻读
事务隔离
扩展:事务的可序列化和实体的可序列化不是一个概念。
实体加序列化是为了可以直接在流中传输实体。
查看事务隔离
show variables like 'tx_isolation'; -- MySQL 5.7之前的版本
show variables like 'transaction_isolation'; -- MySQL 5.7之后的版本
建立表
-- 创建表
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);
-- InnnDB事务自动提交,如果需要演示行锁,需要关闭自动提交
SET autocommit=0;
可重复读,一个事务内,多次读取同一数据是一致的,需要提交当前事务,执行下一次事务。
结论:
如果两个客户端对同一条记录进行修改
- 客户端A修改后,未提交(未commit),此时客户端B修改,则会阻塞
- 客户端A修改后,提交后,客户端B再修改,则不会阻塞
- 如果两个客户端分别对不同的记录行进行修改,则不会被阻塞
3.4索引失效
索引失效,行锁变表锁(通过varchar类型不加单引号让索引失效)
当索引失效后,即使多个客户端操作的不是同一条记录,如果未提交,其他客户端也会进入阻塞状态,所以要避免索引失效
为甚么索引失效行锁会变表锁
- InnoDB 行级锁是通过给索引上的索引项加锁来实现的,InnoDB行级锁只有通过索引条件检索数据,才使用行级锁
- 否则,InnoDB使用表锁 在不通过索引(主 键)条件查询的时候,InnoDB是表锁而不是行锁。
3.5间隙锁
什么是间隙锁
间隙锁危害
间隙锁好处:间隙锁完全解决了幻读。
3.6面试题
3.6.1 如何锁定一行??
优化建议:
优化建议
- 尽可能让数据检索通过索引完成,避免无索引,让行锁升级为表锁
- 合理设计索引,缩小锁的范围
- 尽可能减少检索条件,避免间隙锁
- 尽可能控制事务的大小,减少锁定资源量和时间长度
- 尽可能采用低级别的事务隔离级别
总结:
3.7 什么是页锁?
页锁(补充):
- 开销和加锁时间介于表锁和行锁之间,会出现死锁
- 锁定粒度介于表锁和行锁之间,并发度一般
Mysql之数据库锁(表锁和行锁)详解相关推荐
- MySQL 数据库 User表权限以及用户授权详解
转载:https://blog.mimvp.com/article/27956.html MySQL 常用权限操作 1)本机登陆mysql: mysql -u root -p (-p一定要有):改变数 ...
- 【MySQL】数据库中表的增删查改操作详解
文章目录 前言 SQL的通用语法 一.表的创建与表的新增 语法 数据类型的介绍 演示 二.表的删除 语法 删整张表的语法 删记录的语法 演示 三.表的查询 查询整张表 (一)全列查询 (二)指定列查询 ...
- mysql创建数据库没有默认值报错_详解Mysql数据库date, datetime类型设置0000-00-00默认值(default)报错问题...
现象:MySQL5.7版本之后,date, datetime类型设置默认值"0000-00-00",出现异常:Invalid default value for 'time' 原因 ...
- mysql 并发 锁表_MySQL中的锁(表锁、行锁) 并发控制锁
https://github.com/MrLining/mysql/wiki/MySQL%E4%B8%AD%E7%9A%84%E9%94%81%EF%BC%88%E8%A1%A8%E9%94%81%E ...
- 最全MySQL锁讲解:页锁、共享锁、行锁、表锁、悲观锁、乐观锁
我们在操作数据库的时候,可能会由于并发问题而引起的数据的不一致性(数据冲突),如何保证数据并发访问的一致性.有效性,是所有数据库必须解决的一个问题,锁的冲突也是影响数据库并发访问性能的一个重要因素,从 ...
- mysql行锁怎么读_MySQL锁(三)行锁:幻读是什么?如何解决幻读?
概述 前面两篇文章介绍了MySQL的全局锁和表级锁,今天就介绍一下MySQL的行锁. MySQL的行锁是各个引擎内部实现的,不是所有的引擎支持行锁,例如MyISAM就不支持行锁. 不支持行锁就意味着在 ...
- MySQL获取数据库每个表的行数
这个问题的起因,是我要进行数据库数据的迁移,迁移后确定数据是否都迁移成功而遇到的. 对于数据库的迁移,我是使用MySQL官方的数据库管理工具MySQL Workbench完成的.源数据库和目标数据库都 ...
- java ibatis 锁表_oracle查看被锁的表和解锁
https://www.cnblogs.com/XQiu/p/5212787.html --以下几个为相关表 SELECT * FROM v$lock; SELECT * FROM v$sqlarea ...
- MariaDB/MySQL从数据库中选择随机的行
MariaDB/MySQL从数据库中选择随机的行 一个比较传统的做法是使用sql自带的rand函数,从而达到随机排序的目的. SELECT column FROM table ORDER BY RAN ...
- PHP教程 数据库和MySQL_PHP教程 - MySQL 创建数据库和表
PHP MySQL 创建数据库和表 数据库存有一个或多个表. 创建数据库 CREATE DATABASE 语句用于在 MySQL 中创建数据库. 语法 CREATE DATABASE database ...
最新文章
- java 汇率使用的数据类型_Flink计算支持的数据类型
- STP 简介----生成树算法
- 千万级在线推送系统架构解析
- 用Gradle命令行编译Android工程
- pythonfor循环例句_Python for 循环语句
- urlos 阻止访问ip_URLOS使用流程
- 计算机国二笔试试题,全国计算机等考试二笔试试题(2).ppt
- python中类方法、类实例方法、静态方法的使用与区别
- wlanconnect无法连接wifi_苹果iphone12无法连接wifi怎么回事 解决方法分享
- 温故知新----标签的语义化
- linux下安装使用dig命令
- 【TL431】TL431精密电压调节器简介
- ALS算法的基本思想
- 搜狗云输入法,实现原理.
- gif透明背景动画_在找gif制作app?分享一个GIF制作神器,视频、图片通通可以变GIF...
- skywalking 安装部署以及监控远程应用
- FOLLOW集合求解
- matlab二项式,动态规划 – 计算二项式系统 —MATLAB代码 – 算法网
- 前端学习之HTML入门
- 自增主键的sql设置语句
热门文章
- html5 canvas修改颜色,html5 canvas 笔记二(添加样式和颜色)
- LightOJ 1013 LCS+记忆化搜索
- 【翻译自mos文章】使用aum( Automatic Undo Management) 时遇到 ORA-01555错误--- 原因和解决方式。...
- 浅复制(Shallow Copy)与深复制(Deep Copy)
- ASP.NET2.0快速入门--高级数据方案(3)
- TriCore处理器的上下文切换原理
- Apache 服务器 参数设置
- (83)建立时间与保持时间时序分析技巧
- (50)FPGA状态机描述(三段式)
- (124)FPGA面试题-ZYNQ的PS和PL端怎么交互的?