Atomic分析以及自增问题解决

AtimicXXX

i++

简介

在多线程的情况下,i++不是线程安全的,那么为什么不是线程安全的呢?

i++

在底层会分为三步

内存到寄存器 – 寄存器自增 – 写回内存

  1. 取值
  2. ++操作 i+1;
  3. 赋值 i = i+1

当线程A执行到取值,或者++操作时,线程突然切换到线程B,B修改了值,当线程A再次拿到执行权的时候,值就会改变了。

线程安全:在并发的环境下,拿到的结果和预想的结果不一样就是线程不安全。

实例

public class Test {
    //设置线程数
    private static final int THREADS_CONUT = 20;
    //总数
    public static int count = 0;
    //i++函数
    public static void increase() {
        count++;
    }

    public static void main(String[] args) {
        Thread[] threads = new Thread[THREADS_CONUT];
        for (int i = 0; i < THREADS_CONUT; i++) {
            threads[i] = new Thread(new Runnable() {
                @Override
                public void run() {
                    //每个线程加1000次
                    for (int i = 0; i < 1000; i++) {
                        increase();
                    }
                }
            });
            threads[i].start();
        }
       //Thread.activeCount() 此方法返回活动线程的当前线程的线程组中的数量。
        while (Thread.activeCount() > 2) {
            //保证所有线程都能运行结束
            Thread.yield();
        }
        System.out.println(count);
    }
}

这段程序实现了20个线程,每个线程加1 1000次 所以输出应该是2000,但是输出结果有时会出现偏差,大多数情况下不是2000,而是会比他小,这里就可以看出i++不是线程安全的,那么加上volacitl呢

在public static volatile int count = 0; 给count加上volatile,结果依旧是比20000小的数,这里就可以证明volatial在并发下不是线程安全的,他不保证原子性。

所以问题还是出在自增上。

解决

  1. 加入synchronzied关键字,或者加入lock锁
  2. 使用jdk1.5后java.util.concurrent.atomic包提供了常用的原子操作

AtomicInteger

再来看上面的例子

public class Test {

    private static final int THREADS_CONUT = 20;
    public static AtomicInteger count = new AtomicInteger(0);

    public static void increase() {
        count.incrementAndGet();
    }

    public static void main(String[] args) {
        Thread[] threads = new Thread[THREADS_CONUT];
        for (int i = 0; i < THREADS_CONUT; i++) {
            threads[i] = new Thread(new Runnable() {
                @Override
                public void run() {
                    for (int i = 0; i < 1000; i++) {
                        increase();
                    }
                }
            });
            threads[i].start();
        }

        while (Thread.activeCount() > 2) {
            Thread.yield();
        }
        System.out.println(count);
    }
}

在测试10次后都是20000,输出了正确的结果,说明Atomic具有原子性。

在Java中提供了一些原子的操作类型

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9S4yfgyA-1624676822526)(C:\Users\lenovo\AppData\Roaming\Typora\typora-user-images\image-20210604182916108.png)]

源码分析

首先来看他的属性,可以看到value是由volatile修饰的,保证了内存的可见性

private volatile int value;

重点分析incrementAndGet也就是 i++操作 ****

public final int incrementAndGet() {
    //无限循环
        for (;;) {
            //获取当前值
            int current = get();
            //将+1赋给next
            int next = current + 1;
            //如果比较和next相同,进行set,否则就是被其他线程修改过,重新获取当前值
            if (compareAndSet(current, next))
                return next;
        }
    }
public final boolean compareAndSet(int expect, int update) {
        return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
    }

这里采用的是CAS操作, incrementAndGet() 在一个无限循环体内,不断尝试将一个比自己大1的新值付给自己,如果失败说明在当前值已经被其他线程修改过,于是便再次进入循环下一次操作,直到成功为止。

而compareAndSet()调用的是Unsafe.compareAndSwapInt()方法,即Unsafe类的CAS操作。

这里不懂的可以先去看看乐观锁的介绍

悲观锁和乐观锁

compareAndSwapInt()方法,即Unsafe类的CAS操作。

这里不懂的可以先去看看乐观锁的介绍

悲观锁和乐观锁

Atomic就是volatile的使用场景,也是CAS的使用场景

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值