`
truelove12358
  • 浏览: 77631 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

【MySQL】悲观锁&乐观锁

 
阅读更多

【MySQL】悲观锁&乐观锁概念

悲观锁与乐观锁是两种常见的资源并发锁设计思路,也是并发编程中一个非常基础的概念。本文将对这两种常见的锁机制在数据库数据上的实现进行比较系统的介绍。

悲观锁(Pessimistic Lock)


悲观锁的特点是先获取锁,再进行业务操作,即“悲观”的认为获取锁是非常有可能失败的,因此要先确保获取锁成功再进行业务操作。通常所说的“一锁二查三更新”即指的是使用悲观锁。通常来讲在数据库上的悲观锁需要数据库本身提供支持,即通过常用的select … for update操作来实现悲观锁。当数据库执行select for update时会获取被select中的数据行的行锁,因此其他并发执行的select for update如果试图选中同一行则会发生排斥(需要等待行锁被释放),因此达到锁的效果。select for update获取的行锁会在当前事务结束时自动释放,因此必须在事务中使用。

这里需要注意的一点是不同的数据库对select for update的实现和支持都是有所区别的,例如oracle支持select for update no wait,表示如果拿不到锁立刻报错,而不是等待,mysql就没有no wait这个选项。另外mysql还有个问题是select for update语句执行中所有扫描过的行都会被锁上,这一点很容易造成问题。因此如果在mysql中用悲观锁务必要确定走了索引,而不是全表扫描。

乐观锁(Optimistic Lock)


乐观锁的特点先进行业务操作,不到万不得已不去拿锁。即“乐观”的认为拿锁多半是会成功的,因此在进行完业务操作需要实际更新数据的最后一步再去拿一下锁就好。

乐观锁在数据库上的实现完全是逻辑的,不需要数据库提供特殊的支持。一般的做法是在需要锁的数据上增加一个版本号,或者时间戳,然后按照如下方式实现:

复制代码
1. SELECT data AS old_data, version AS old_version FROM …;
2. 根据获取的数据进行业务操作,得到new_data和new_version
3. UPDATE SET data = new_data, version = new_version WHERE version = old_version
if (updated row > 0) {
    // 乐观锁获取成功,操作完成
} else {
    // 乐观锁获取失败,回滚并重试
}
复制代码

乐观锁是否在事务中其实都是无所谓的,其底层机制是这样:在数据库内部update同一行的时候是不允许并发的,即数据库每次执行一条update语句时会获取被update行的写锁,直到这一行被成功更新后才释放。因此在业务操作进行前获取需要锁的数据的当前版本号,然后实际更新数据时再次对比版本号确认与之前获取的相同,并更新版本号,即可确认这之间没有发生并发的修改。如果更新失败即可认为老版本的数据已经被并发修改掉而不存在了,此时认为获取锁失败,需要回滚整个业务操作并可根据需要重试整个过程。

总结

  • 乐观锁在不发生取锁失败的情况下开销比悲观锁小,但是一旦发生失败回滚开销则比较大,因此适合用在取锁失败概率比较小的场景,可以提升系统并发性能

  • 乐观锁还适用于一些比较特殊的场景,例如在业务操作过程中无法和数据库保持连接等悲观锁无法适用的地方

参考资料Internal Locking Methods


乐观锁 优缺点

乐观锁:

乐观锁( Optimistic Locking ) 相对悲观锁而言,乐观锁机制采取了更加宽松的加锁机制。悲观锁大多数情况下依靠数据库的锁机制实现,以保证操作最大程度的独占性。但随之而来的就是数据库性能的大量开销,特别是对长事务而言,这样的开销往往无法承受。而乐观锁机制在一定程度上解决了这个问题。乐观锁,大多是基于数据版本( Version )记录机制实现。何谓数据版本?即为数据增加一个版本标识,在基于数据库表的版本解决方案中,一般是通过为数据库表增加一个 “version” 字段来实现。读取出数据时,将此版本号一同读出,之后更新时,对此版本号加一。更新数据时将读出的版本号作为一个条件,如果数据库表中版本号与此版本号相同则能更新成功,否则更新失败。

例如一个金融系统,当某个操作员读取用户的数据,并在读出的用户数据的基础上进行修改时(如更改用户帐户余额),如果采用悲观锁机制,也就意味着整个操作过 程中(从操作员读出数据、开始修改直至提交修改结果的全过程,甚至还包括操作 员中途去煮咖啡的时间),数据库记录始终处于加锁状态,可以想见,如果面对几百上千个并发,这样的情况将导致怎样的后果。

乐观锁机制在一定程度上解决了这个问题。乐观锁,大多是基于数据版本 ( Version )记录机制实现。何谓数据版本?即为数据增加一个版本标识,在基于数据库表的版本解决方案中,一般是通过为数据库表增加一个 “version” 字段来实现(也可以采用另一种方式,同样是在需要乐观锁控制的table中增加一个字段,名称无所谓,字段类型使用时间戳timestamp, 和上面的version类似,也是在更新的时候检查当前数据库中数据的时间戳和自己更新前取到的时间戳进行对比,如果一致则OK,否则就是版本冲突)。

优点:

从上面的例子可以看出,乐观锁机制避免了长事务中的数据库加锁开销,大大提升了大并发量下的系统整体性能表现。

缺点:

乐观锁机制往往基于系统中的数据存储逻辑,因此也具备一定的局限性,如在上例中,由于乐观锁机制是在我们的系统中实现,来自外部系统的更新操作不受我们系统的控制,因此可能会造成脏数据被更新到数据库中。在系统设计阶段,应该充分考虑到这些情况出现的可能性,并进行相应调整(如将乐观锁策略在数据库存储过程中实现,对外只开放基于此存储过程的数据更新途径,而不是将数据库表直接对外公开)。


使用实例:


MySQL中的隔离级别和悲观锁及乐观锁示例

1,MySQL的事务支持
MySQL的事务支持不是绑定在MySQL服务器本身,而是与存储引擎相关:

  1. MyISAM:不支持事务,用于只读程序提高性能
  2. InnoDB:支持ACID事务、行级锁、并发
  3. BerkeleyDB:支持事务


2,隔离级别
隔离级别决定了一个session中的事务可能对另一个session的影响、并发session对数据库的操作、一个session中所见数据的一致性
ANSI标准定义了4个隔离级别,MySQL的InnoDB都支持:

Java代码

  1. READUNCOMMITTED:最低级别的隔离,通常又称为dirtyread,它允许一个事务读取还没commit的数据,这样可能会提高性能,但是dirtyread可能不是我们想要的
  2. READCOMMITTED:在一个事务中只允许已经commit的记录可见,如果session中select还在查询中,另一session此时insert一条记录,则新添加的数据不可见
  3. REPEATABLEREAD:在一个事务开始后,其他session对数据库的修改在本事务中不可见,直到本事务commit或rollback。在一个事务中重复select的结果一样,除非本事务中update数据库。
  4. SERIALIZABLE:最高级别的隔离,只允许事务串行执行。为了达到此目的,数据库会锁住每行已经读取的记录,其他session不能修改数据直到前一事务结束,事务commit或取消时才释放锁。

可以使用如下语句设置MySQL的session隔离级别:

[c-sharp]view plaincopy
  1. 1.SETTRANSACTIONISOLATIONLEVEL{READUNCOMMITTED|READCOMMITTED|REPEATABLEREAD|SERIALIZABLE}

MySQL默认的隔离级别是REPEATABLE READ,在设置隔离级别为READ UNCOMMITTED或SERIALIZABLE时要小心,READ UNCOMMITTED会导致数据完整性的严重问题,而SERIALIZABLE会导致性能问题并增加死锁的机率

3,隔离级别
乐观所和悲观锁策略:
悲观锁:在读取数据时锁住那几行,其他对这几行的更新需要等到悲观锁结束时才能继续
乐观所:读取数据时不锁,更新时检查是否数据已经被更新过,如果是则取消当前更新
一般在悲观锁的等待时间过长而不能接受时我们才会选择乐观锁
悲观锁的例子:

[c-sharp]view plaincopy
  1. CREATEPROCEDUREtfer_funds
  2. (from_accountINT,to_accountINT,tfer_amountNUMERIC(10,2),
  3. OUTstatusINT,OUTmessageVARCHAR(30))
  4. BEGIN
  5. DECLAREfrom_account_balanceNUMERIC(10,2);
  6. STARTTRANSACTION;
  7. SELECTbalance
  8. INTOfrom_account_balance
  9. FROMaccount_balance
  10. WHEREaccount_id=from_account
  11. FORUPDATE;
  12. IFfrom_account_balance>=tfer_amountTHEN
  13. UPDATEaccount_balance
  14. SETbalance=balance-tfer_amount
  15. WHEREaccount_id=from_account;
  16. UPDATEaccount_balance
  17. SETbalance=balance+tfer_amount
  18. WHEREaccount_id=to_account;
  19. COMMIT;
  20. SETstatus=0;
  21. SETmessage='OK';
  22. ELSE
  23. ROLLBACK;
  24. SETstatus=-1;
  25. SETmessage='Insufficientfunds';
  26. ENDIF;
  27. END;

乐观锁的例子:

[c-sharp]view plaincopy
  1. CREATEPROCEDUREtfer_funds
  2. (from_accountINT,to_accountINT,tfer_amountNUMERIC(10,2),
  3. OUTstatusINT,OUTmessageVARCHAR(30))
  4. BEGIN
  5. DECLAREfrom_account_balanceNUMERIC(8,2);
  6. DECLAREfrom_account_balance2NUMERIC(8,2);
  7. DECLAREfrom_account_timestamp1TIMESTAMP;
  8. DECLAREfrom_account_timestamp2TIMESTAMP;
  9. SELECTaccount_timestamp,balance
  10. INTOfrom_account_timestamp1,from_account_balance
  11. FROMaccount_balance
  12. WHEREaccount_id=from_account;
  13. IF(from_account_balance>=tfer_amount)THEN
  14. --Hereweperformsomelongrunningvalidationthat
  15. --mighttakeafewminutes*/
  16. CALLlong_running_validation(from_account);
  17. STARTTRANSACTION;
  18. --Makesuretheaccountrowhasnotbeenupdatedsince
  19. --ourinitialcheck
  20. SELECTaccount_timestamp,balance
  21. INTOfrom_account_timestamp2,from_account_balance2
  22. FROMaccount_balance
  23. WHEREaccount_id=from_account
  24. FORUPDATE;
  25. IF(from_account_timestamp1<>from_account_timestamp2OR
  26. from_account_balance<>from_account_balance2)THEN
  27. ROLLBACK;
  28. SETstatus=-1;
  29. SETmessage=CONCAT("Transactioncancelledduetoconcurrentupdate",
  30. "ofaccount",from_account);
  31. ELSE
  32. UPDATEaccount_balance
  33. SETbalance=balance-tfer_amount
  34. WHEREaccount_id=from_account;
  35. UPDATEaccount_balance
  36. SETbalance=balance+tfer_amount
  37. WHEREaccount_id=to_account;
  38. COMMIT;
  39. SETstatus=0;
  40. SETmessage="OK";
  41. ENDIF;
  42. ELSE
  43. ROLLBACK;
  44. SETstatus=-1;
  45. SETmessage="Insufficientfunds";
  46. ENDIF;
  47. END$$








分享到:
评论

相关推荐

    mysql 悲观锁与乐观锁的理解及应用分析

    本文实例讲述了mysql 悲观锁与乐观锁。分享给大家供大家参考,具体如下: 悲观锁与乐观锁是人们定义出来的概念,你可以理解为一种思想,是处理并发资源的常用手段。 不要把他们与mysql中提供的锁机制(表锁,行锁,...

    MySQL中的悲观锁与乐观锁

    在MySQL数据库中,悲观锁(Pessimistic Lock)和乐观锁(Optimistic Lock)是两种常见的并发控制机制,它们用于解决多用户环境下同一资源的并发访问问题。这两种锁各有特点,适用于不同的业务场景。 首先,悲观锁...

    Mysql悲观锁和乐观锁的使用示例

    MySQL提供了两种常见的锁策略:悲观锁和乐观锁,它们各有特点,适用于不同的场景。下面我们将详细探讨这两种锁的工作原理以及如何在MySQL中实现它们。 **悲观锁(Pessimistic Lock)** 悲观锁正如其名,它假设数据...

    mysql的乐观锁、悲观锁.md

    java,乐观锁,悲观锁详解释

    悲观锁和乐观锁.md

    所谓悲观锁,总是假设最坏的情况,每次去拿数据的时候都会认为别人会修改数据,造成幻读,不可重复读,脏读等情况发生。所谓乐观锁,重视假设最好的情况,每次去拿数据都认为别人不会修改,所以不会上锁,但是会在...

    36谈谈MySQL支持的事务隔离级别,以及悲观锁和乐观锁的原理和应用场景?

    36谈谈MySQL支持的事务隔离级别,以及悲观锁和乐观锁的原理和应用场景?

    SpringBoot整合MyBatis实现乐观锁和悲观锁的示例

    SpringBoot整合MyBatis实现乐观锁和悲观锁的示例 在本文中,我们将学习如何使用SpringBoot和MyBatis来实现乐观锁和悲观锁。我们将通过示例代码来介绍这两种锁的实现方式,帮助读者更好地理解和使用它们。 一、悲观...

    Hibernate实现悲观锁和乐观锁代码介绍

    Hibernate 实现悲观锁和乐观锁代码介绍 Hibernate 是一个基于 Java 的持久层框架,它提供了多种锁机制来实现事务的隔离性和一致性。在本文中,我们将详细介绍 Hibernate 实现悲观锁和乐观锁的代码实现,并讨论 ...

    实例讲解MySQL中乐观锁和悲观锁

    乐观锁和悲观锁式并发控制主要采用的技术手段 悲观锁 在关系数据库管理系统中,悲观并发控制(悲观锁,PCC)是一种并发控制的方法。它可以阻止一个事务以影响其他用户的方式来修改数据。如果一个事务执行的操作的每...

    各种锁汇总,乐观锁、悲观锁、分布式锁、可重入锁、互斥锁、读写锁、分段锁、类锁、行级锁等

    本文将深入探讨标题和描述中提及的各种锁,包括乐观锁、悲观锁、分布式锁、可重入锁、互斥锁、读写锁、分段锁、类锁以及行级锁。 1. **乐观锁**:乐观锁假设多线程环境中的冲突较少,所以在读取数据时不加锁,只有...

    MySQL锁机制,乐观锁,悲观锁等

    锁的定义:   数据库锁定机制简单来说,就是数据库为了保证数据的一致性,而使各种共享资源在被并发访问变得有序所设计的一种规则。 表级锁:开销小,加锁快;...一、悲观锁   顾名思义,就是对于数

    乐观锁悲观锁及事务及行锁表锁

    mysql锁,与事务,以及各种级别锁,和乐观锁悲观锁的研究使用

    MySQL面试题(记得被问过的一些题目)

    什么是乐观锁和悲观锁? 乐观锁和悲观锁的区别? 乐观锁和悲观锁的使用场景? 什么是死锁? 解决死锁的机制有哪些? 发生死锁的场景有哪些? 什么是事务? 事务的特性与适用场景? 事务的隔离级别与适用场景?

    并发编程下的锁机制,乐观锁、悲观锁、共享锁、排他锁、分布式锁、锁降级原理篇

    本文将详细讲解几种常见的锁机制:悲观锁、乐观锁、共享锁和排他锁,并简要介绍分布式锁以及锁降级原理。 1. **悲观锁**: 悲观锁是一种保守的策略,它假设在读取数据时,数据极有可能被其他线程修改。因此,悲观...

Global site tag (gtag.js) - Google Analytics