热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

对于Mysql中锁的理解

Mysql数据库中锁的概念非常多,这次我就来分享一下我对于Mysql中锁的理解 首先我们先明确一下锁的概念: 锁是计算机协调进程或线程并发访问某一资源的机制,在数据库中要保证数据并发访问的一致性、

Mysql数据库中锁的概念非常多,这次我就来分享一下我对于Mysql中锁的理解
首先我们先明确一下锁的概念:
锁是计算机协调进程或线程并发访问某一资源的机制,在数据库中要保证数据并发访问的一致性、有效性。锁冲突也是影响数据库并发性能的一个重要因素。
我们再来看两个名词:

  • 乐观锁:每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在提交更新的时候会判断一下在此期间别人有没有去更新这个数据。(在数据进行提交更新的时候,才会正式对数据的冲突与否进行检测,如果发现冲突了,则返回给用户错误的信息,让用户决定如何去做)
  • 悲观锁:每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会阻止,直到这个锁被释放。(在修改数据之前先锁定,再修改的方式被称之为悲观并发控制)

我们肯定听说过,Mysql中有表锁和行锁之分,那么表锁和行锁到底是什么呢?

  • 表级锁:开销小,加锁快;不会产生死锁;锁定粒度大,发生锁冲突的概率最高;并发度最低
  • 行级锁:开销大,加锁慢;会产生死锁;锁定粒度小,发生锁冲突的概率最低;并发度最高
  • 页面锁:开销和加锁时间界于表级锁和行级锁之间;会产生死锁;锁定粒度界于表解锁与行级锁之间;并发度一般

    来看看MyISAM的表锁,MyISAM的表锁分为共享读锁(Table Read Lock)和独占写锁(Table Write Lock)

    对于Mysql中锁的理解
    MyISAM的表锁.png

    我们通过SQL的形式分别来看看这两种锁到底是怎么运作的

    首先是共享读锁:
    在同一个session中

-- 加读锁
LOCK TABLE my_account READ
-- 查询自己的这一张表不会报错
SELECT * FROM my_account
-- 加了读锁,在同一个session中对表进行修改会报错
UPDATE my_account SET mOney= money - 50 WHERE id = 1
-- 对另外的表进行写操作(报错)
UPDATE test SET c1 = 1 WHERE id = 1
-- 对另外的表进行读操作(报错)
SELECT * FROM test
-- 使用别名查询同一张表也会报错(此处只能查my_account)
SELECT a.* FROM my_account a
-- 若一定要使用该别名则在加锁时就设置别名
LOCK TABLE my_account AS a READ
-- 释放锁
UNLOCK TABLES

在另一个session

-- 在另一个session中对表进行修改会等待锁释放,锁释放后就执行
UPDATE my_account SET mOney= money - 50 WHERE id = 1
-- 在另一个session中对另一个表进行操作不会报错,正常执行

然后是独占写锁
在同一个session中

-- 加写锁
LOCK TABLE my_account WRITE
-- 对同一张表进行操作不会报错
INSERT INTO my_account VALUES(3,3,'wangwu',0)
DELETE FROM my_account WHERE id = 3
SELECT * FROM my_account
-- 对其他表进行查询(报错)
SELECT * FROM test 
-- 对其他表进行修改(报错)
INSERT INTO test VALUES(10,10,10)
-- 释放锁
UNLOCK TABLES

在另一个session

-- 在另一张表中对该表进行操作需要等待写锁释放
SELECT * FROM my_account
INSERT INTO my_account VALUES(3,3,'wangwu',0)
DELETE FROM my_account WHERE id = 3

总结一下:

  • 对于MyISAM表的读锁,不会阻塞当前用户对表的读请求,但对写请求会报错
  • 对于MyISAM表的读锁,不会阻塞其他用户对同一表的读请求,但会阻塞对同一表的写请求
  • 一个session使用LOCK TABLE命令给表f加了读锁,这个session可以查询锁定表中的日志,但对其他表进行访问和修改都会报错
  • 对于MyISAM表的写锁,会阻塞其他用户对同一表的读写操作
  • 对于MyISAM表的写锁,当前用户可以对本表进行读写操作,但对其他表进行操作会报错

再来看看Inoodb的行锁

  • 共享锁:又称读锁,当一个事务对某几行上读锁时,允许其他事务对这几行进行读操作,但不允许进行写操作。允许其他事务给这几行上读锁,但不允许上写锁。
  • 排它锁:又称写锁,当一个事务对某几行上写锁时,不允许其他事务写,但允许读,更不允许其他事务给这几行上任何锁。

共享锁

BEGIN
-- 上一个行读锁
SELECT * FROM test WHERE id = 1 LOCK IN SHARE MODE 
-- 提交事务
COMMIT
-- 对于被锁的这一行的写操作要等待读锁的释放
UPDATE test SET c1 = '1' WHERE id = 1
-- 读操作正常执行
SELECT * FROM test WHERE id = 1

排它锁

BEGIN
-- 上一个行写锁
SELECT * FROM test WHERE id = 1 FOR UPDATE
-- 等待写锁的释放
UPDATE test SET c1 = '1' WHERE id = 1
-- 正常执行
UPDATE test SET c1 = '1' WHERE id = 2
-- 读操作正常执行
SELECT * FROM test WHERE id = 1

总结一下:

  • 共享锁:又称读锁,当一个事务对某几行上读锁时,允许其他事务对这几行进行读操作,但不允许进行写操作。允许其他事务给这几行上读锁,但不允许上写锁。
  • 排它锁:又称写锁,当一个事务对某几行上写锁时,不允许其他事务写,但允许读,更不允许其他事务给这几行上任何锁。

那么什么情况下行锁会升级成表锁?

  • 如果没有索引,更新数据(update、delete)时会锁住整张表。
    InnoDB 行级锁是通过给索引上的索引项加锁来实现的,InnoDB行级锁只有通过索引条件检索数据,才使用行级锁;否则,InnoDB使用表锁 在不通过索引(主 键)条件查询的时候,InnoDB是表锁而不是行锁。

  • 事务隔离级别为:串行化时,读写数据都会锁住整张表。(一次只能一个人操作表)

Mysql中还有间隙锁,什么是间隙锁,什么方式会产生间隙锁?
概念:当我们用范围条件而不是相等条件检索数据,并请求共享或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个“间隙”加锁,这种锁机制就是所谓的间隙锁(NEXT-KEY)锁。

假设有以下表my_account:(其中id为PK,id和name为联合索引)

对于Mysql中锁的理解
间隙锁1.png
BEGIN
SELECT id,name FROM my_account WHERE number > 0 AND number 

所有数据都会命中,而因为命中了索引,加的是行锁。而此时就会产生间隙锁对于键值在条件范围内但不存在的记录,叫做“间隙(GAP)**
在另一个session

BEGIN
INSERT INTO my_account VALUES(5,5,'test',0)

插入id=5的数据,是间隙,被上了间隙锁,因此需要等待上一个事务结束以后才会执行


推荐阅读
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 本文详细介绍了MysqlDump和mysqldump进行全库备份的相关知识,包括备份命令的使用方法、my.cnf配置文件的设置、binlog日志的位置指定、增量恢复的方式以及适用于innodb引擎和myisam引擎的备份方法。对于需要进行数据库备份的用户来说,本文提供了一些有价值的参考内容。 ... [详细]
  • 本文由编程笔记小编整理,介绍了PHP中的MySQL函数库及其常用函数,包括mysql_connect、mysql_error、mysql_select_db、mysql_query、mysql_affected_row、mysql_close等。希望对读者有一定的参考价值。 ... [详细]
  • 如何在php中将mysql查询结果赋值给变量
    本文介绍了在php中将mysql查询结果赋值给变量的方法,包括从mysql表中查询count(学号)并赋值给一个变量,以及如何将sql中查询单条结果赋值给php页面的一个变量。同时还讨论了php调用mysql查询结果到变量的方法,并提供了示例代码。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • PHP设置MySQL字符集的方法及使用mysqli_set_charset函数
    本文介绍了PHP设置MySQL字符集的方法,详细介绍了使用mysqli_set_charset函数来规定与数据库服务器进行数据传送时要使用的字符集。通过示例代码演示了如何设置默认客户端字符集。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • Oracle分析函数first_value()和last_value()的用法及原理
    本文介绍了Oracle分析函数first_value()和last_value()的用法和原理,以及在查询销售记录日期和部门中的应用。通过示例和解释,详细说明了first_value()和last_value()的功能和不同之处。同时,对于last_value()的结果出现不一样的情况进行了解释,并提供了理解last_value()默认统计范围的方法。该文对于使用Oracle分析函数的开发人员和数据库管理员具有参考价值。 ... [详细]
  • 高质量SQL书写的30条建议
    本文提供了30条关于优化SQL的建议,包括避免使用select *,使用具体字段,以及使用limit 1等。这些建议是基于实际开发经验总结出来的,旨在帮助读者优化SQL查询。 ... [详细]
  • 本文介绍了Windows操作系统的版本及其特点,包括Windows 7系统的6个版本:Starter、Home Basic、Home Premium、Professional、Enterprise、Ultimate。Windows操作系统是微软公司研发的一套操作系统,具有人机操作性优异、支持的应用软件较多、对硬件支持良好等优点。Windows 7 Starter是功能最少的版本,缺乏Aero特效功能,没有64位支持,最初设计不能同时运行三个以上应用程序。 ... [详细]
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
author-avatar
小文982_412
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有