重庆分公司,新征程启航

为企业提供网站建设、域名注册、服务器等服务

什么是悲观锁

什么是悲观锁?针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。

成都创新互联公司为企业级客户提高一站式互联网+设计服务,主要包括成都网站设计、成都网站制作、重庆App定制开发小程序设计、宣传片制作、LOGO设计等,帮助客户快速提升营销能力和企业形象,创新互联各部门都有经验丰富的经验,可以确保每一个作品的质量和创作周期,同时每年都有很多新员工加入,为我们带来大量新的创意。 

悲观锁指的是对数据被外界修改持保守态度,就是在数据处理过程中,将数据处于锁定状态,该功能需要依靠数据库提供的锁机制,否则即使在系统中实现了加锁机制,也无法保证外部系统不会修改数据。

锁的模式

LockMode.NONE

无锁机制

LockMode.READ

Hibernate在读取记录时自动获取锁

即共享锁:(Shared lock, S 锁),共享锁又称读锁。如果事务 T 获得了数据对象 A 上的共享锁(也就是说对 A 加上共享锁),那么其他事务只能获得 A 上的 共享锁(S 锁),而不能加排他锁(X 锁),直到 A 释放所有的共享锁。获准共享锁的事务只能读数据,不能修改数据。

LockMode.WRITE

Hibernate在insert获者update记录时自动获取锁

即排他锁:(Exclusive lock, X 锁),排他锁又称写锁。如果事务 T 获得了数据 A 上的排他锁,那么 T 既可以读又可以写 A,但是在 T 释放 A 上的 X 锁之前,其他事务既不能获得 A 上的共享锁,也不能获得 A 上的排他锁。

LockMode.UPGRADE

如果数据库系统支持悲观锁(如Oracle和MySQL),就执行select…for update语句(行级锁住,其他事务不能对其进行update、insert和delete语句),如果数据库不支持悲观锁(如Sybase),就执行普通的select语句。

LockMode.UPGRADE_NOWAIT

和LockMode.UPGRADE具有相同的功能。此外,对于Oracle数据库执行select…for update nowait语句。”nowait”表示如果执行该select语句的事务不能立刻获得悲观锁,那么不会等待其他事务释放锁,而是立刻抛出一个锁定异常。

锁的演示

注意:锁只对一次事务中操作的数据对象起作用,并不是对整个数据库起作用而将整个数据库锁住。

打开两个SQL命令操作行界面,这两个界面可以代表两个事务 T1 和 T2。我们先在两个界面中分别执行命令:start transaction;

在事务 T1 中在数据库中查询学号为 “2015” 的学生信息并对操作的数据加上悲观锁:select * from stu_info where stu_no=”2015” for update;。此时我们能查询到对应的信息。

“for update” 就表示加上悲观锁。此次查询完并不提交事务,也就是说还未释放锁

在事务 T2 中在数据库中查询学号为 “2016” 的学生信息并对操作的数据加上悲观锁:select * from stu_info where stu_no=”2016” for update;。此时我们也能查询到对应的信息。

尚不提交事务

在事务 T2 中在数据库中查询学号为 “2015” 的学生信息,此次做普通的 select 查询,不加锁:select * from stu_info where stu_no=”2015”;。此时我们也能查询到对应的信息。

尚不提交事务

在事务 T2 中尝试查询数据库中学号为 “2015” 的学生信息并对操作的数据加上悲观锁:select * from stu_info where stu_no=”2015” for update;。此时我们能发现,并未显示相关的信息,而是在等待中。当我们提交事务 T1 时(即:commit),数据对象 “2015” 释放 T1 中的悲观锁,T2 事务才能查询到相关的信息并获取到悲观锁。

如果我们换一下步骤5。在刚才的第五步中,我们在事务 T1 中修改学号为 “2016”学生信息但是不加悲观锁,因为此前我们在 事务 T2 中对该记录加了悲观锁还未释放,所以,只有等 T2 提交后才能修改成功。

关于什么是悲观锁问题的解答就分享到这里了,希望以上内容可以对大家有一定的帮助,如果你还有很多疑惑没有解开,可以关注创新互联行业资讯频道了解更多相关知识。


文章标题:什么是悲观锁
分享链接:http://cqcxhl.cn/article/pjihph.html

其他资讯

在线咨询
服务热线
服务热线:028-86922220
TOP