Java并发编程之原子操作类

2018-10-2321:23:31后端程序开发Comments2,872 views字数 1618阅读模式

原子操作类简介文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

当更新一个变量的时候,多出现数据争用的时候可能出现所意想不到的情况。这时的一般策略是使用synchronized解决,因为synchronized能够保证多个线程不会同时更新该变量。然而,从jdk 5之后,提供了粒度更细、量级更轻,并且在多核处理器具有高性能的原子操作类。因为原子操作类把竞争的范围缩小到单个变量上,这可以算是粒度最细的情况了。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

原子操作类相当于泛化的volatile变量,能够支持原子读取-修改-写操作。比如AtomicInteger表示一个int类型的数值,提供了get和set方法,这些volatile类型的变量在读取与写入上有着相同的内存语义。原子操作类共有13个类,在java.util.concurrent.atomic包下,可以分为四种类型的原子更新类:原子更新基本类型、原子更新数组类型、原子更新引用和原子更新属性。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

下面将分别介绍这四种原子操作类。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

原子更新基本类型文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

使用原子方式更新基本类型,共包括3个类:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicBoolean:原子更新布尔变量文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicInteger:原子更新整型变量文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicLong:原子更新长整型变量文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

具体到每个类的源代码中,提供的方法基本相同,这里以AtomicInteger为例进行说明。AtomicInteger提供的部分方法如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

为了说明AtomicInteger的原子性,这里代码演示多线程对一个int值进行自增操作,最后输出结果,代码如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

输出结果如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

可以看到在多线程的情况下,得到的结果是正确的,但是如果仅仅使用int类型的成员变量则可能得到不同的结果。这里的关键在于getAndIncrement是原子操作,那么是如何保证的呢?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

getAndIncrement方法的源码如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

到这里可以发现最终调用了native方法来保证更新的原子性。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

原子更新数组文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

通过原子更新数组里的某个元素,共有3个类:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicIntegerArray:原子更新整型数组的某个元素文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicLongArray:原子更新长整型数组的某个元素文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicReferenceArray:原子更新引用类型数组的某个元素文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicIntegerArray常用的方法有:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

int addAndSet(int i, int delta):以原子方式将输入值与数组中索引为i的元素相加文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

boolean compareAndSet(int i, int expect, int update):如果当前值等于预期值,则以原子方式更新数组中索引为i的值为update值文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

示例代码如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

运行结果是:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

数组value通过构造的方式传入AtomicIntegerArray中,实际上AtomicIntegerArray会将当前数组拷贝一份,所以在数组拷贝的操作不影响原数组的值。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

原子更新引用类型文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

需要更新引用类型往往涉及多个变量,早atomic包有三个类:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicReference:原子更新引用类型文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicReferenceFieldUpdater:原子更新引用类型里的字段文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicMarkableReference:原子更新带有标记位的引用类型。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

下面以AtomicReference为例进行说明:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

可以看到user被成功更新。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

原子更新字段类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

如果需要原子更新某个类的某个字段,就需要用到原子更新字段类,可以使用以下几个类:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicIntegerFieldUpdater:原子更新整型字段文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicLongFieldUpdater:原子更新长整型字段文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

AtomicStampedReference:原子更新带有版本号的引用类型。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

要想原子更新字段,需要两个步骤:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

每次必须使用newUpdater创建一个更新器,并且需要设置想要更新的类的字段文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

更新类的字段(属性)必须为public volatile文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

下面的代码演示如何使用原子更新字段类更新字段:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

输出的结果如下:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

Java并发编程之原子操作类文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

至此,我们知道了如何使用原子操作类在不同场景下的基本用法。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/bc/7089.html

  • 本站内容整理自互联网,仅提供信息存储空间服务,以方便学习之用。如对文章、图片、字体等版权有疑问,请在下方留言,管理员看到后,将第一时间进行处理。
  • 转载请务必保留本文链接:https://www.cainiaoxueyuan.com/bc/7089.html

Comment

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定