Atomic原子操作类介绍

在并发编程中很容易出现并发安全的问题,有一个很简单的例子就是多线程更新变量i=1,比 如多个线程执行i++操作,就有可能获取不到正确的值,而这个问题,最常用的方法是通过 Synchronized进行控制来达到线程安全的目的。但是由于synchronized是采用的是悲观锁策 略,并不是特别高效的一种解决方案。实际上,在J.U.C下的atomic包提供了一系列的操作简单, 性能高效,并能保证线程安全的类去更新基本类型变量,数组元素,引用类型以及更新对象中的 字段类型。atomic包下的这些类都是采用的是乐观锁
摘要由CSDN通过智能技术生成

在并发编程中很容易出现并发安全的问题,有一个很简单的例子就是多线程更新变量i=1,比 如多个线程执行i++操作,就有可能获取不到正确的值,而这个问题,最常用的方法是通过 Synchronized进行控制来达到线程安全的目的。但是由于synchronized是采用的是悲观锁策 略,并不是特别高效的一种解决方案。实际上,在J.U.C下的atomic包提供了一系列的操作简单, 性能高效,并能保证线程安全的类去更新基本类型变量,数组元素,引用类型以及更新对象中的 字段类型。atomic包下的这些类都是采用的是乐观锁策略去原子更新数据,在java中则是使用 CAS操作具体实现。

在java.util.concurrent.atomic包里提供了一组原子操作类:

基本类型:AtomicInteger、AtomicLong、AtomicBoolean;

引用类型:AtomicReference、AtomicStampedRerence、AtomicMarkableReference;

数组类型:AtomicIntegerArray、AtomicLongArray、AtomicReferenceArray

对象属性原子修改器:AtomicIntegerFieldUpdater、AtomicLongFieldUpdater、 AtomicReferenceFieldUpdater

原子类型累加器(jdk1.8增加的类):DoubleAccumulator、DoubleAdder、 LongAccumulator、LongAdder、Striped64

原子更新基本类型

以AtomicInteger为例总结常用的方法

1 //以原子的方式将实例中的原值加1,返回的是自增前的旧值;
2 public final int getAndIncrement() {
3 return unsafe.getAndAddInt(this, valueOffset, 1);
4 }
5
6 //getAndSet(int newValue):将实例中的值更新为新值,并返回旧值;
7 public final boolean getAndSet(boolean newValue) {
8 boolean prev;
9 do {
10 prev = get();
11 } while (!compareAndSet(prev, newValue));
12 return prev;
13 }
14
15 //incrementAndGet() :以原子的方式将实例中的原值进行加1操作,并返回最终相加后的结
果;
16 public final int incrementAndGet() {
17 return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
18 }
19
20 //addAndGet(int delta) :以原子方式将输入的数值与实例中原本的值相加,并返回最后的结
果;
21 public final int addAndGet(int delta) {
22 return unsafe.getAndAddInt(this, valueOffset, delta) + delta;

测试

1 public class AtomicIntegerTest {
2 static AtomicInteger sum = new AtomicInteger(0);
3
4 public static void main(String[] args) {
5
6 for (int i = 0; i < 10; i++) {
7 Thread thread = new Thread(() ‐> {
8 for (int j = 0; j < 10000; j++) {
9 // 原子自增 CAS
10 sum.incrementAndGet();
11 //TODO
12 }
13 });
14 thread.start();
15 }
16
17 try {
18 Thread.sleep(3000);
19 } catch (InterruptedException e) {
20 e.printStackTrace();
21 }
22 System.out.println(sum.get());
23
24 }
25
26 }

incrementAndGet()方法通过CAS自增实现,如果CAS失败,自旋直到成功+1。

 原子更新数组类型

AtomicIntegerArray为例总结常用的方法

 

1 //addAndGet(int i, int delta):以原子更新的方式将数组中索引为i的元素与输入值相加;
2 public final int addAndGet(int i, int delta) {
3 return getAndAdd(i, delta) + delta;
4 }
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值