什么是ABA问题?Java里面的原生解决方案是什么?原理是什么?

开发 后端
AtomicStampedReference是一个带有时间戳的对象引用,能很好的解决CAS机制中的ABA问题,这篇文章将通过案例对其介绍分析。

[[381390]]

 AtomicStampedReference是一个带有时间戳的对象引用,能很好的解决CAS机制中的ABA问题,这篇文章将通过案例对其介绍分析。

一、ABA问题

ABA问题是CAS机制中出现的一个问题,他的描述是这样的。我们直接画一张图来演示,

 

什么意思呢?就是说一个线程把数据A变为了B,然后又重新变成了A。此时另外一个线程读取的时候,发现A没有变化,就误以为是原来的那个A。这就是有名的ABA问题。ABA问题会带来什么后果呢?我们举个例子。

一个小偷,把别人家的钱偷了之后又还了回来,还是原来的钱吗,ABA问题也一样,如果不好好解决就会带来大量的问题。最常见的就是资金问题,也就是别人如果挪用了你的钱,在你发现之前又还了回来。但是别人却已经触犯了法律。

如何去解决这个ABA问题呢,就是使用今天所说的AtomicStampedReference。

二、AtomicStampedReference

1、问题解决

我们先给出一个ABA的例子,对ABA问题进行场景重现。

  1. public class AtomicTest { 
  2.  private static AtomicInteger index = new AtomicInteger(10); 
  3.  public static void main(String[] args) { 
  4.   new Thread(() -> { 
  5.    index.compareAndSet(10, 11); 
  6.    index.compareAndSet(11, 10); 
  7.    System.out.println(Thread.currentThread().getName()+ 
  8.      ":10->11->10"); 
  9.   },"张三").start(); 
  10.    
  11.   new Thread(() -> { 
  12.    try { 
  13.     TimeUnit.SECONDS.sleep(2); 
  14.     boolean isSuccess = index.compareAndSet(10, 12); 
  15.     System.out.println(Thread.currentThread().getName()+ 
  16.       ":index是预期的10嘛,"+isSuccess 
  17.       +"   设置的新值是:"+index.get()); 
  18.    } catch (InterruptedException e) { 
  19.     e.printStackTrace(); 
  20.    } 
  21.   },"李四").start(); 
  22.  } 

在上面的代码中,我们使用张三线程,对index10->11->10的变化,然后李四线程读取index观察是否有变化,并设置新值。运行一下看看结果:

 

这个案例重现了ABA的问题场景,下面我们看如何使用AtomicStampedReference解决这个问题的。

  1. public class AtomicTest2 { 
  2.  private static AtomicInteger index = new AtomicInteger(10); 
  3.  static AtomicStampedReference<Integer> stampRef  
  4.        = new AtomicStampedReference(10, 1); 
  5.  public static void main(String[] args) { 
  6.   new Thread(() -> { 
  7.    int stamp = stampRef.getStamp(); 
  8.    System.out.println(Thread.currentThread().getName()  
  9.      + " 第1次版本号: " + stamp); 
  10.    stampRef.compareAndSet(10, 11,stampRef.getStamp(),stampRef.getStamp()+1); 
  11.    System.out.println(Thread.currentThread().getName()  
  12.      + " 第2次版本号: " + stampRef.getStamp()); 
  13.    stampRef.compareAndSet(11, 10,stampRef.getStamp(),stampRef.getStamp()+1); 
  14.    System.out.println(Thread.currentThread().getName()  
  15.      + " 第3次版本号: " + stampRef.getStamp()); 
  16.   },"张三").start(); 
  17.    
  18.   new Thread(() -> { 
  19.    try { 
  20.     int stamp = stampRef.getStamp(); 
  21.     System.out.println(Thread.currentThread().getName()  
  22.       + " 第1次版本号: " + stamp); 
  23.     TimeUnit.SECONDS.sleep(2); 
  24.     boolean isSuccess =stampRef.compareAndSet(10, 12, 
  25.       stampRef.getStamp(),stampRef.getStamp()+1); 
  26.     System.out.println(Thread.currentThread().getName()  
  27.       + " 修改是否成功: "+ isSuccess+" 当前版本 :" + stampRef.getStamp()); 
  28.     System.out.println(Thread.currentThread().getName()  
  29.       + " 当前实际值: " + stampRef.getReference()); 
  30.    } catch (InterruptedException e) { 
  31.     e.printStackTrace(); 
  32.    } 
  33.   },"李四").start(); 
  34.  } 

上面的代码我们再来分析一下,我们会发现AtomicStampedReference里面增加了一个时间戳,也就是说每一次修改只需要设置不同的版本好即可。我们先运行一边看看:

 

这里使用的是AtomicStampedReference的compareAndSet函数,这里面有四个参数:

compareAndSet(V expectedReference, V newReference, int expectedStamp, int newStamp)。

(1)第一个参数expectedReference:表示预期值。

(2)第二个参数newReference:表示要更新的值。

(3)第三个参数expectedStamp:表示预期的时间戳。

(4)第四个参数newStamp:表示要更新的时间戳。

这个compareAndSet方法到底是如何实现的,我们深入到源码中看看。

2、源码分析

  1. public boolean compareAndSet(V   expectedReference, 
  2.                                  V   newReference, 
  3.                                  int expectedStamp, 
  4.                                  int newStamp) { 
  5.         Pair<V> current = pair; 
  6.         return 
  7.             expectedReference == current.reference && 
  8.             expectedStamp == current.stamp && 
  9.             ((newReference == current.reference && 
  10.               newStamp == current.stamp) || 
  11.              casPair(current, Pair.of(newReference, newStamp))); 
  12.     } 

刚刚这四个参数的意思已经说了,我们主要关注的就是实现,首先我们看到的就是这个Pair,因此想要弄清楚,我们再看看这个Pair是什么,

  1. private static class Pair<T> { 
  2.      final T reference; 
  3.      final int stamp; 
  4.      private Pair(T reference, int stamp) { 
  5.          this.reference = reference; 
  6.          this.stamp = stamp; 
  7.      } 
  8.      static <T> Pair<T> of(T reference, int stamp) { 
  9.          return new Pair<T>(reference, stamp); 
  10.      } 
  11.  } 

在这里我们会发现Pair里面只是包存了值reference和时间戳stamp。

在compareAndSet方法中最后还调用了casPair方法,从名字就可以看到,主要是使用CAS机制更新新的值reference和时间戳stamp。我们可以进入这个方法中看看。

  1. //底层调用的是UNSAFE的compareAndSwapObject方法 
  2.  
  3.  private boolean casPair(Pair<V> cmp, Pair<V> val) { 
  4.      return UNSAFE.compareAndSwapObject(this, pairOffset, cmp, val); 
  5.  } 

三、总结

其实除了AtomicStampedReference类,还有一个原子类也可以解决,就是AtomicMarkableReference,它不是维护一个版本号,而是维护一个boolean类型的标记,用法没有AtomicStampedReference灵活。因此也只是在特定的场景下使用。

 本文转载自微信公众号「愚公要移山」,可以通过以下二维码关注。转载本文请联系愚公要移山公众号。

 

责任编辑:武晓燕 来源: 愚公要移山
相关推荐

2021-02-02 18:02:09

java对象数据

2024-11-19 17:54:15

JavaCASABA问题

2021-02-02 09:37:20

CQRS系统数据库

2024-11-25 12:20:00

Hystrix微服务架构

2011-06-30 18:23:08

2017-03-21 23:29:44

DevOps运维开发

2023-09-26 07:57:17

C#.NetCCtor函数

2023-04-20 16:55:22

区块链数组货币

2024-02-04 00:01:00

云原生技术容器

2023-11-05 10:52:54

DNS服务器浏览器

2021-07-23 10:11:33

物联网IOT

2024-06-24 00:07:00

开源es搜索引擎

2024-01-11 08:53:58

2020-06-08 12:47:08

ETLHadoopSpark

2020-11-05 09:47:48

云计算IT技术

2020-12-17 17:40:01

物联网智能IOT

2019-07-29 07:29:05

2009-07-07 16:50:39

ServletResp

2009-09-16 09:39:50

ccna是什么CCNA

2021-09-27 08:02:17

CDN加速网站网络
点赞
收藏

51CTO技术栈公众号