乐观锁以及乐观锁的实现「建议收藏」

乐观锁以及乐观锁的实现「建议收藏」乐观锁介绍: 乐观锁( Optimistic Locking ) 相对悲观锁而言,乐观锁假设认为数据一般情况下不会造成冲突,所以在数据进行提交更新的时候,才会正式对数据的冲突与否进行检测,如果发现冲…

乐观锁以及乐观锁的实现

乐观锁介绍:

乐观锁( Optimistic Locking ) 相对悲观锁而言,乐观锁假设认为数据一般情况下不会造成冲突,所以在数据进行提交更新的时候,才会正式对数据的冲突与否进行检测,如果发现冲突了,则让返回用户错误的信息,让用户决定如何去做。那么我们如何实现乐观锁呢,一般来说有以下2种方式:

1.使用数据版本(Version)记录机制实现,这是乐观锁最常用的一种实现方式。何谓数据版本?即为数据增加一个版本标识,一般是通过为数据库表增加一个数字类型的 “version” 字段来实现。当读取数据时,将version字段的值一同读出,数据每更新一次,对此version值加一。当我们提交更新的时候,判断数据库表对应记录的当前版本信息与第一次取出来的version值进行比对,如果数据库表当前版本号与第一次取出来的version值相等,则予以更新,否则认为是过期数据。用下面的一张图来说明:

如上图所示,如果更新操作顺序执行,则数据的版本(version)依次递增,不会产生冲突。但是如果发生有不同的业务操作对同一版本的数据进行修改,那么,先提交的操作(图中B)会把数据version更新为2,当A在B之后提交更新时发现数据的version已经被修改了,那么A的更新操作会失败。

2.乐观锁定的第二种实现方式和第一种差不多,同样是在需要乐观锁控制的table中增加一个字段,名称无所谓,字段类型使用时间戳(timestamp), 和上面的version类似,也是在更新提交的时候检查当前数据库中数据的时间戳和自己更新前取到的时间戳进行对比,如果一致则OK,否则就是版本冲突。

一、为什么需要锁(并发控制)?
      在多用户环境中,在同一时间可能会有多个用户更新相同的记录,这会产生冲突。这就是著名的并发性问题。
      典型的冲突有:
        1.丢失更新:一个事务的更新覆盖了其它事务的更新结果,就是所谓的更新丢失。例如:用户A把值从6改为2,用户B把值从2改为6,则用户A丢失了他的更新。
        2.脏读:当一个事务读取其它完成一半事务的记录时,就会发生脏读取。例如:用户A,B看到的值都是6,用户B把值改为2,用户A读到的值仍为6。

       为了解决这些并发带来的问题。 我们需要引入并发控制机制。

二、 并发控制机制

             锁,即给我们选定的目标数据上锁,使其无法被其他程序修改。

            1.悲观锁:指的是对数据被外界(包括本系统当前的其他事务,以及来自外部系统的事务处理)修改持保守态度,因此,在整个数据处理过程中,将数据处于锁定状态
            2.乐观锁:假设不会发生并发冲突,只在提交操作时检查是否违反数据完整性。乐观锁不能解决脏读的问题。

三、乐观锁的实现
        使用数据版本(Version)记录机制实现,这是乐观锁最常用的一种实现方式。何谓数据版本?即为数据增加一个版本标识,一般是通过为数据库表增加一个数字类型的 “version” 字段来实现。当读取数据时,将version字段的值一同读出,数据每更新一次,对此version值加一。当我们提交更新的时候,判断数据库表对应记录的当前版本信息与第一次取出来的version值进行比对,如果数据库表当前版本号与第一次取出来的version值相等,则予以更新,否则认为是过期数据

   1.数据库表设计

     task

   有三个字段,分别是id,value、version

   2.实现

   1)先读task表的数据(实际上这个表只有一条记录),得到version的值为versionValue

   2)每次更新task表中的value字段时,为了防止发生冲突,需要这样操作

      update task set value = newValue,version =  versionValue + 1   where version = versionValue;

      只有这条语句执行了,才表明本次更新value字段的值成功

如假设有两个节点A和B都要更新task表中的value字段值,差不多在同一时刻,A节点和B节点从task表中读到的version值为2,那么A节点和B节点在更新value字段值的时候,都操作 update task set value = newValue,version =  3   where version = 2;,实际上只有1个节点执行该SQL语句成功,假设A节点执行成功,那么此时task表的version字段的值是3,B节点再操作update task set value = newValue,version =  3   where version = 2;这条SQL语句是不执行的,这样就保证了更新task表时不发生冲突

四、项目中使用案例

    /**
     * 基于乐观锁的更新操作
     * @param editFinance 编辑的账户对象
     * @param queryLockNo 上次查询的乐观锁版本号
     * @return
     */
    @Override
    public int updateForLockNo(BzFinanceEntity editFinance, int queryLockNo) {
 
        editFinance.setLockNo(queryLockNo + 1); //修改乐观锁版本
 
        BzFinanceEntityExample example = new BzFinanceEntityExample();
        BzFinanceEntityExample.Criteria criteria = example.createCriteria();
        criteria.andIdFinanceEqualTo(editFinance.getIdFinance());
        criteria.andLockNoEqualTo(queryLockNo); //基于乐观锁,修改查询版本的数据
        
        //根据Example条件更新实体BzFinanceEntity包含的不是null的属性值
        int mark = this.baseEntityDao.updateByExampleSelective(editFinance, example);
 
        return mark;
    }
 
————————————————
版权声明:本文为CSDN博主「Anyhowe」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/sunwenhao_2017/java/article/details/81565783

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/8028.html

(0)
上一篇 2023-03-09 09:30
下一篇 2023-03-09

相关推荐

  • 计算机语言中的奇妙旋律——Python sin of 5

    计算机语言中的奇妙旋律——Python sin of 5Python语言中的sin()函数是计算给定角度的正弦值,而sin of 5则是指将角度设置为5度时的sin()值。Python中的sin()函数是math库中的函数,可以使用from math import sin来导入该函数。

    2024-03-16
    74
  • python学习等疑惑(为什么学python原因)

    python学习等疑惑(为什么学python原因) Python是一种跨平台的计算机程序设计语言,是一种面向对象的动态类型语言,越来越多被用于独立的,大型项目的开发,已被逐渐广泛应用于系统管理任务的处理和Web编程。下面给大家带来一些关于Python 学习心得 ,希望对大家有所帮助。

    2023-11-19
    134
  • 【Linux系列】Centos 7安装 Redis(六)「建议收藏」

    【Linux系列】Centos 7安装 Redis(六)「建议收藏」目的 本文主要介绍以下两点: 一. 安装Redis 二. 设置开机启动项 演示 一. 安装Redis 打开 ‘Redis官网’ ,右击复制链接。 启动redis 启动成功 二. 设置开机启动项 到/e

    2022-12-22
    159
  • Python判断空值方法

    Python判断空值方法Python是一门广泛应用于数据科学和人工智能领域的编程语言。在开发过程中,我们会遇到许多需要判断空值的情况。本文将从多个方面介绍Python中判断空值的方法。

    2024-06-25
    49
  • Linux的基础——虚拟机的克隆

    Linux的基础——虚拟机的克隆1.虚拟机的安装 虚拟机的安装在另一个文档 安装jdk(在另外一个文档中) 2.虚拟机的克隆 准备工作:一台装有Linux系统的主机(已经配置好jdk) 选择主机进行克隆 注意:这里一定要选择创建完整

    2023-04-21
    147
  • Python正则表达式应用实例:匹配邮箱地址

    Python正则表达式应用实例:匹配邮箱地址在现代社会中,邮箱是人们日常生活和工作中必不可少的电子邮件工具,因此如何正确而高效地匹配各种类型的邮箱地址,成为了学习使用Python的正则表达式的必要基础内容。

    2024-02-07
    93
  • Python卸载方法

    Python卸载方法Python是一种高级编程语言,常用于数据分析、人工智能、Web开发等领域。随着Python技术的普及,越来越多的人开始学习和使用Python。然而,在某些情况下,你可能需要卸载Python,比如需要换用另一种编程语言,或者需要重新安装Python程序等。本文将介绍Python卸载的方法。

    2024-07-31
    33
  • 使用Python ttk来实现GUI界面快速开发

    使用Python ttk来实现GUI界面快速开发Graphical User Interface, 简称 GUI,是指采用图形方式显示应用程序的程序界面,用户可以通过鼠标、键盘等与应用程序进行直接交互。

    2024-01-14
    107

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注