文章目录

  • Java 乐观锁和悲观锁
    • 1、悲观锁
    • 2、乐观锁
      • 2.1 CAS
      • 2.2 模拟CAS算法
      • 2.3 JUC
      • 2.4 CAS中的ABA问题
      • 2.5 使用CAS会引发的问题

Java 乐观锁和悲观锁

1、悲观锁

总是假设最坏的情况,每次在去获取共享数据的时候都认为别人会修改,所以每次都在获取数据的时候加锁。 传统的关系型数据库里就用到很多这种锁,比如行锁,表锁、读锁、写锁等都是在操作之前先上锁,比如java中Synchronized关键字的实现也是悲观锁。

悲观锁存在的问题

在多线程竞争下,加锁,释放锁会导致比较多的上下文切换和调度延迟,引起性能问题一个线程持有锁会导致其他需要此锁的线程挂起

2、乐观锁

认为数据一般情况下不会产生并发冲突,所以在数据进行提交更新的时候,才会对数据是否产生并发冲突进行检测,如果发现并发冲突,则返回错误信息,需要用户去决定如何操作。乐观锁实现的典型是CAS(Compare and Swap)

2.1 CAS

具有原子特性
CAS乐观锁的实现技术,当多个线程尝试使用CAS同时更新同一个变量,只有一个线程能更新变量的值,而其他的线程都失败,失败的线程不会被挂起,而是被告知这次竞争失败了,并可以再次进行尝试。

CAS操作中涉及三个操作数:
● 需要读写的内存位置(V)
● 需要比较的预期原值(A)
● 拟写入的新值(B)

如果内存位置V的值与预期原值A相匹配,那么处理器会自动的将该位置值更新为B,否则处理器不做任何处理。
第一步: 获取位置V的值A
第二步: 将获取的值A和位置V的内容进行比较,如果相等,认为没有其他线程修改该位置,即不存在并发竞争,就可以将新值B写入位置V。如果不相等,说明存在其他的线程在对该位置进行并发操作。不能直接修改,继续跳转第一步,获取位置V的值,再进行比较,直至相等再修改为B。

2.2 模拟CAS算法

CompareAndSwap类


public class CompareAndSwap {private int value;//获取内存值public synchronized int getValue() {return value;}//比较并交换public synchronized int compareAndSwap(int expectedValue, int newValue) {int oldValue = value;//如果内存值和预期值一致,就替换if (oldValue == expectedValue) {this.value = newValue;}return oldValue;}//设置调用比较并交换, 看期望值和原来的值是否一致public synchronized boolean compareAndSet(int expectValue, int newValue) {return expectValue == compareAndSwap(expectValue, newValue);}
}

TestCAS类

import java.util.Random;public class TestCAS {public static void main(String[] args) {CompareAndSwap compareAndSwap = new CompareAndSwap();for (int i = 0; i < 10; i++) {new Thread(new Runnable() {@Overridepublic void run() {Random random = new Random();int expectedValue = compareAndSwap.getValue();int newValue = random.nextInt(100);boolean b = compareAndSwap.compareAndSet(expectedValue, newValue);System.out.println("线程:" + Thread.currentThread().getName() + ",预期值:" + expectedValue + ",待写入值:"  + newValue + ",操作结果:" + b);}}).start();}}
}

运行结果:

2.3 JUC

在JDK 1.5中新增了java.util.concurrent(J.U.C)建立在CAS之上,相对于Synchronized是一种线程阻塞处理,CAS是非阻塞的一种常见实现,及线程即使没有获取到变量,也不会进入到阻塞状态。就是在不使用锁的情况下来保证线程安全,在JUC下存在如AtomicInteger为例,其中一些++i操作是安全性操作,如getAndIncrement方法。

代码示例:

public class TestJUC {public static void main(String[] args) {AtomicInteger atomicInteger = new AtomicInteger();for (int i = 0; i < 10; i++)new Thread(new Runnable() {@Overridepublic void run() {int i = atomicInteger.getAndIncrement();System.out.println(Thread.currentThread().getName() +",预期值:" + i);}}).start();}
}

运行结果:

2.4 CAS中的ABA问题

CAS会引起ABA的问题,假如存在如下执行序列:

1、线程1从内存中V取出A
2、线程2从内存中V取出A
3、线程2进行了一些操作,将B写入位置V。
4、线程2将A再次写入位置V
5、线程1进行CAS操作,发现位置V依然是A,进行修改操作并成功
6、尽管线程1的CAS操作成功,但不代表这个过程没有问题,对于线程1,线程2的修改已经丢失了。

一个链表ABA的问题:

1、 现有一个单向链表实现的堆栈,栈顶为A。这时线程1已经知道A.next是B,希望通过CAS操作将栈顶替换为B,线程1执行compareAndSwap(A,B)
2、 在线程1执行上面指令之前,线程2介入,将A、B出栈,在依次入栈D、C、A,而对象B次数处于游离状态。
3、 此时线程1执行CAS操作,检测栈顶认为A,所以CAS成功,栈顶是B,但实际B.next为null,此时堆栈中只有一个B元素,C和D组成的链表就不存在在堆栈中,C、D被丢弃了

ABA问题的解决

ABA问题的解决思路就是使用版本号,在变量上追加一个版本号,每次变量变更把版本号加1,那么A-B-A就回去变成1A-2B-3A

2.5 使用CAS会引发的问题

使用CAS好处就是被使用锁的开销要小,但存在问题

  • ABA的问题
    ABA的问题的解决方案是加版本号解决
  • 循环时间开销大
    如果CAS如果长时间不成功,会给CPU带来非常大的执行开销
  • 只能保证一个共享变量的原子操作
    当对一个共享变量执行操作时,可以使用循环CAS的方式保证原子操作,但对于多个共享变量,循环CAS就无法保证操作的原子性,这个时候就需要借助于锁来实现

Java 乐观锁和悲观锁相关推荐

  1. **Java有哪些悲观锁的实现_面试4连问:乐观锁与悲观锁的概念、实现方式、场景、优缺点?...

    推荐阅读: 数据库面试4连问:分库分表,中间件,优缺点,如何拆分? 终极手撕之架构大全:分布式+框架+微服务+性能优化,够不够? 消息队列面试,你能顶得住面试官这波10大连环炮的攻势吗? 01 乐观锁 ...

  2. 详解各种锁:CAS、共享锁、排它锁、互斥锁、悲观锁、乐观锁、行级锁、表级锁、页级锁、死锁、JAVA对CAS的支持、ABA问题、AQS原理

    共享锁(S锁) 又称为读锁,可以查看但无法修改和删除的一种数据锁.如果事务T对数据A加上共享锁后,则其他事务只能对A再加共享锁,不能加排它锁.获准共享锁的事务只能读数据,不能修改数据. 共享锁下其它用 ...

  3. Java并发篇_乐观锁与悲观锁

    乐观锁对应于生活中乐观的人总是想着事情往好的方向发展,悲观锁对应于生活中悲观的人总是想着事情往坏的方向发展. 一.引入概念 1.悲观锁 总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次 ...

  4. Java多线程学习总结(5)——乐观锁和悲观锁的基本概念、实现方式(含实例)、适用场景及常见面试题

     分享一个大神的人工智能教程.零基础!通俗易懂!风趣幽默!还带黄段子!希望你也加入到人工智能的队伍中来!点击浏览教程 一.基本概念 乐观锁和悲观锁是两种思想,用于解决并发场景下的数据竞争问题. 乐观锁 ...

  5. Java并发 乐观锁和悲观锁 乐观锁的一种实现方式CAS

    为什么80%的码农都做不了架构师?>>>    首先介绍一些乐观锁与悲观锁: 悲观锁:总是假设最坏的情况,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人 ...

  6. Java之乐观锁和悲观锁

    Java之乐观锁和悲观锁 乐观锁 用到的机制是CAS(Compare and Swap),每个线程都可以访问,只有在提交数据的时候,检查是否违反了数据的完整性.如果发生冲突失败重试,直到成功为止.乐观 ...

  7. Java并发问题--乐观锁与悲观锁以及乐观锁的一种实现方式-CAS

    Java并发问题–乐观锁与悲观锁以及乐观锁的一种实现方式-CAS </h1><div class="clear"></div><div c ...

  8. java中的锁(悲观锁、乐观锁、可重入锁、不可重入锁、公平锁、非公平锁、自旋锁、阻塞锁...)

    Lock接口 1.简介.地位.作用 ① 锁是一种工具,用于控制对共享资源的访问 ② Lock和synchronized,这两个是最常见的锁,它们都可以达到线程安全的目的,但是在使用和功能上又有较大的不 ...

  9. java中的锁---乐观锁与悲观锁的区别

    锁,是开发中不得不掌握的一个知识点. 在面试中也会经常问到.其中乐观锁与悲观锁为最常见.首先介绍下两种锁 一:乐观锁: 1 介绍: 它的心态很好,每次别人使用它的时候,它会乐观的认为别人不修改数据,所 ...

  10. java 乐观锁和悲观锁,Threadlocal

    7.乐观锁和悲观锁 悲观锁(Pessimistic Lock)顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会 block 直到它拿到 ...

最新文章

  1. React 16.8.6 发布,构建用户界面的 JavaScript 库
  2. C++中数字和字符的转换
  3. Codeforces Round #631 (Div. 2) D. Dreamoon Likes Sequences 思维 + 组合数学
  4. 【原创】PSP开机只是绿灯亮,黑屏
  5. Redisbook学习笔记(1)字典(2)
  6. 模仿showModalDialog的总在最前
  7. Shiro面试题(二十道)
  8. smart原则_设立目标的smart原则
  9. Process Monitor工具找网吧广告
  10. 自动驾驶各大传感器介绍-硬件篇
  11. php的网页服务器根目录,php获得网站根目录的几个方法
  12. 简单学JAVA-Java前世今生
  13. [转]人生就像一张茶几,摆满了各种杯具洗具餐具
  14. mysql如何锁表和解锁
  15. Sql Server备份时提示:备份文件不可用,原先扇区为512,现在为4096
  16. centos7下安装nginx
  17. win10美化工具全套详细解析
  18. DJBCTF2021-WEB-虎山行
  19. 七.Elasticsearch 字段折叠
  20. 【计网复习】第三章 数据链路层

热门文章

  1. java开发环境的配置总结_后端实习1:Java开发环境配置处理
  2. OpenGL MSAA:抗锯齿技术详解
  3. mysql数据库的授权访问
  4. (个人经历)实习生春招面试中的TCP/IP常见(安全)问题及其相应解答
  5. FL Studio2023水果音乐制作入门教程
  6. 原生JavaScript实现显示当前系统时间
  7. 土壤中氮含量的测定分析
  8. schtasks命令创建计划任务
  9. 两个Vector交集、并集、补集运算
  10. Ubantu 16.04 LTS版本映像下载