`
森林的天空
  • 浏览: 15245 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

JAVA并发编程学习笔记之CAS操作

 
阅读更多

 

CAS操作

CAS是单词compare and set的缩写,意思是指在set之前先比较该值有没有变化,只有在没变的情况下才对其赋值。

我们常常做这样的操作

 

  1. if(a==b) {  
  2.     a++;  
  3. }  

试想一下如果在做a++之前a的值被改变了怎么办?a++还执行吗?出现该问题的原因是在多线程环境下,a的值处于一种不定的状态。采用锁可以解决此类问题,但CAS也可以解决,而且可以不加锁。

 

  1. int expect = a;  
  2. if(a.compareAndSet(expect,a+1)) {  
  3.     doSomeThing1();  
  4. else {  
  5.     doSomeThing2();  
  6. }  

这样如果a的值被改变了a++就不会被执行。

按照上面的写法,a!=expect之后,a++就不会被执行,如果我们还是想执行a++操作怎么办,没关系,可以采用while循环

 

  1. while(true) {  
  2.     int expect = a;  
  3.     if (a.compareAndSet(expect, a + 1)) {  
  4.         doSomeThing1();  
  5.         return;  
  6.     } else {  
  7.         doSomeThing2();  
  8.     }  
  9. }  

采用上面的写法,在没有锁的情况下实现了a++操作,这实际上是一种非阻塞算法。

 

应用

java.util.concurrent.atomic包中几乎大部分类都采用了CAS操作,以AtomicInteger为例,看看它几个主要方法的实现:

 

  1. public final int getAndSet(int newValue) {  
  2.     for (;;) {  
  3.         int current = get();  
  4.         if (compareAndSet(current, newValue))  
  5.             return current;  
  6.     }  
  7. }  

getAndSet方法JDK文档中的解释是:以原子方式设置为给定值,并返回旧值。原子方式体现在何处,就体现在compareAndSet上,看看compareAndSet是如何实现的:

  1. public final boolean compareAndSet(int expect, int update) {  
  2.     return unsafe.compareAndSwapInt(this, valueOffset, expect, update);  
  3. }  

不出所料,它就是采用的Unsafe类的CAS操作完成的。


 


再来看看a++操作是如何实现的:

  1. public final int getAndIncrement() {  
  2.     for (;;) {  
  3.         int current = get();  
  4.         int next = current + 1;  
  5.         if (compareAndSet(current, next))  
  6.             return current;  
  7.     }  
  8. }  

几乎和最开始的实例一模一样,也是采用CAS操作来实现自增操作的。


++a操作和a++操作类似,只不过返回结果不同罢了

  1. public final int incrementAndGet() {  
  2.     for (;;) {  
  3.         int current = get();  
  4.         int next = current + 1;  
  5.         if (compareAndSet(current, next))  
  6.             return next;  
  7.     }  
  8. }  

此外,java.util.concurrent.ConcurrentLinkedQueue类全是采用的非阻塞算法,里面没有使用任何锁,全是基于CAS操作实现的。CAS操作可以说是JAVA并发框架的基础,整个框架的设计都是基于CAS操作的。

 

缺点:

1、ABA问题

CAS操作容易导致ABA问题,也就是在做a++之间,a可能被多个线程修改过了,只不过回到了最初的值,这时CAS会认为a的值没有变。a在外面逛了一圈回来,你能保证它没有做任何坏事,不能!!也许它讨闲,把b的值减了一下,把c的值加了一下等等,更有甚者如果a是一个对象,这个对象有可能是新创建出来的,a是一个引用呢情况又如何,所以这里面还是存在着很多问题的,解决ABA问题的方法有很多,可以考虑增加一个修改计数,只有修改计数不变的且a值不变的情况下才做a++,也可以考虑引入版本号,当版本号相同时才做a++操作等,这和事务原子性处理有点类似!

2、比较花费CPU资源,即使没有任何争用也会做一些无用功。

3、会增加程序测试的复杂度,稍不注意就会出现问题。

 

总结:

可以用CAS在无锁的情况下实现原子操作,但要明确应用场合,非常简单的操作且又不想引入锁可以考虑使用CAS操作,当想要非阻塞地完成某一操作也可以考虑CAS。不推荐在复杂操作中引入CAS,会使程序可读性变差,且难以测试,同时会出现ABA问题。

 

参考资料:

分享到:
评论

相关推荐

    Java并发编程学习笔记

    Java并发编程是Java开发中必不可少的一部分,涉及到多线程、同步机制、线程池以及并发工具类等多个核心知识点。以下是对这些主题的详细说明: 1. **线程安全与锁 Synchronized 底层实现原理**: 线程安全是指在多...

    Java并发编程学习笔记.

    Java并发编程是Java开发中的重要领域,它涉及到多线程、同步、锁机制、线程池等关键概念,是提高程序性能和效率的关键技术。在Java中,并发编程的运用可以充分利用多核处理器的能力,实现高效的多任务处理。以下是对...

    Java并发编程与高并发解决方案笔记-基础篇.docx

    Java并发编程与高并发解决方案是开发高性能应用的关键技术。在基础篇中,主要涉及以下几个重要知识点: 1. **并发编程基础** - **并发**:并发是指在一个时间段内,多个线程交替执行,使得系统看起来像是同时处理...

    图灵Java高级互联网架构师第6期并发编程专题笔记.zip

    06-并发编程之CAS&Atomic原子操作详解-fox 07-并发锁机制之深入理解synchronized(一)-fox 08-并发锁机制之深入理解synchronized(二)-fox 09-深入理解AQS之独占锁ReentrantLock源码分析-fox 10-深入理解AQS之...

    并发编程之一 日常学习笔记

    本篇笔记主要关注并发编程中的两个关键概念:CAS(Compare and Swap)原子操作和Java线程的深入理解。 首先,我们来详细探讨一下CAS(比较并交换)原子操作。CAS是一种无锁算法,它通过比较内存中的某个值与期望值...

    Java 并发编程学习笔记之Synchronized底层优化

    Java并发编程中的Synchronized是Java实现线程同步的关键机制,其在JDK1.6之后进行了大量的优化,包括引入了轻量级锁和偏向锁,以提升并发性能。以下是关于这些优化的详细解释: **一、重量级锁** 重量级锁是基于...

    Java分布式应用学习笔记06浅谈并发加锁机制分析

    ### Java分布式应用学习笔记06浅谈并发加锁机制分析 #### 1. 前言 在深入探讨Java中的并发加锁机制之前,我们有必要回顾一下多线程环境下的一些基本...希望本文能帮助读者更好地理解和掌握Java并发编程的相关知识。

    Java很好的学习笔记4 无锁.md,学习代码

    在Java编程领域,无锁编程是一种高级的并发控制技术,旨在提高多线程环境下的程序性能和可伸缩...学习这些内容,有助于开发者深入理解Java并发编程,提升在多线程环境下的编程能力,从而设计出更加高效、可扩展的系统。

    尚硅谷JUC视频笔记整理,很详细和全面,帮你迅速掌握JUC

    JUC(Java Util Concurrent),即Java的并发工具包,是Java提供的一套并发编程解决方案,它通过一系列...通过学习JUC中的关键知识点,我们可以更好地理解Java并发编程原理,并在实际工作中解决多线程编程中遇到的问题。

    Java分布式应用学习笔记03JVM对线程的资源同步和交互机制

    ### Java分布式应用学习笔记03:JVM对线程的资源同步和交互机制 在深入探讨Java虚拟机(JVM)如何处理线程间的资源同步与交互机制之前,我们先来明确几个关键概念:线程、多线程、同步、并发以及它们在Java中的实现...

    java并发框架源码-notes:记录各种学习笔记(Java、算法、框架、数据库、并发、源码...)

    通过阅读和研究这些源码,开发者不仅可以提升对Java并发编程的理解,还能学习到如何在实际项目中应用这些知识,优化代码性能,解决并发问题。日常总结和记录这些知识点,是提升个人技术能力的有效途径。

    Java架构面试专题(含答案)和学习笔记-1

    3. **并发编程**: - **线程与进程**:线程的概念、状态转换、同步与互斥,以及线程池的使用。 - **并发工具类**:如Semaphore、CountDownLatch、CyclicBarrier、ThreadPoolExecutor等的使用。 - **并发原语**:...

    JavaHub:Java程序员学习之路,持续更新原创内容,欢迎Star

    高并发编程-CAS深入解析 高并发编程-CyclicBarrier深入解析 高并发编程-Semaphore深入解析 高并发编程-ReentrantLock公平锁深入解析 高并发编程-ReentrantLock非公平锁深入解析 高并发编程-Reent

    Java多线程源码笔记.pdf

    10. Atomic:Java的原子类如AtomicInteger、AtomicLong等,提供了基于CAS(Compare and Swap)操作的原子性更新,保证了在高并发环境下的数据一致性。 11. ThreadLocal:ThreadLocal为每个线程提供了一个独立的变量...

    java.util.concurrent 测试源文件

    本测试源文件主要是针对JUC并发编程的笔记、API理解和原子类CAS的实践。 首先,让我们深入了解`java.util.concurrent`包的核心概念: 1. **Executor框架**:这是JUC包的基础,提供了一种将任务提交给线程池的统一...

    尚硅谷大厂面试题第三季周阳主讲

    【标题】"尚硅谷大厂面试题第三季周阳主讲"主要涵盖了Java后端面试中的核心知识点,包括Java并发编程(JUC)、Redis缓存系统以及Spring框架的应用。本内容整理了B站尚硅谷频道关于大厂面试的全部资料,帮助考生全面...

    Lock free 论文集合,若干无锁数据结构实现的经典论文,500多页.zip

    在描述中提到了“大学生 C/C++/JAVA/Python数据结构学习笔记和资料大全”,这表明除了无锁编程的论文,还可能包含广泛的数据结构学习资源。数据结构是计算机科学的基础,对于理解和优化算法至关重要。C/C++通常用于...

    锁机制学习与总结笔记参考

    在计算机科学和编程领域,尤其是多线程编程中,锁机制是确保数据一致性与线程安全的关键工具。本文将深入探讨锁的原理、类型以及在实际应用中的使用方法。 一、锁的概述 锁机制是一种同步机制,用于控制对共享资源...

Global site tag (gtag.js) - Google Analytics