Java原子变量类常见问题解决

网友投稿 239 2022-12-12


Java原子变量类常见问题解决

在学习多线程时,遇到了原子变量类,它是基于 CAS 和 volatile 实现的,能够保障对共享变量进行 read-modify-write 更新操作的原子性和可见性。于是我就写了一段代码试试,自认为非常正确。

public class Test{

private static AtomicInteger ID = new AtomicInteger(0);

public static int nextID(){ //返回的ID范围为 1~100

http://if(ID.get() == 100) { //ID到达100时,则从1开始

ID.set(1);

return ID.get(); // return ID = 1;

}

else

return ID.incrementAndGet(); //++ID

}

public static void main(String[] args) throws Exception{

for(int i = 0; i < 5; i++){

new Thread(()->{

for(int j = 0; j < 100; j++)

nextID();

}).start();

}

Thread.sleep(1000); //应该输出100才对

System.out.println(ID);

}

}

用五个线程并发获得ID,每个线程获取100个,最后应该输出100才是,但试了好几次都不是100。原子变量类不是能保障原子性和可见性吗,为什么出现了竞态?

纠结了很久,还是很懵逼。后来发现 get 方法相当于读取一个 volatile 变量,而读取一个 volatile 变量时,不具备排他性!(AtomicInteger类内部使用了volatile修饰了value值,而volatile关键字不具备排他性)

也就是说,当一个线程刚读取到了共享的 volatile 变量的值时,其他线程可会马上对共享变量进行修改。如,线程A读取到ID的值为99时(还没对ID进行修改),其他线程可能马上就将ID加1了,此时共享变量为100了,其他线程再获取ID时,应该令ID=1才是,但线程A已经进入了else分支,它还认为ID=99,而不知道其他线程刚把ID加1变成了100,所以会吧ID加上1变成了101,这就出现了竞态。

对于同一个共享变量而言,一个线程更新了该变量的值之后,其他线程能够读取到这个更新后的值,那这个值就被称为该变量的 相对新值。

如果读取这个共享变量的线程在读取并使用该变量的时候其他线程无法更新该变量的值,那么该线程读取到的相对新值就被称为该变量的 最新值。需要加锁,才能读取到最新值。

解决办法,使用原子操作 compareAndSet:

private static int nextID(){ //返回的ID范围为 1~100

ID.compareAndSet(100, 0);

return ID.increhttp://mentAndGet();

}


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:使用idea搭建一个spring mvc项目的图文教程
下一篇:SpringBoot2整合ElasticJob框架过程详解
相关文章

 发表评论

暂时没有评论,来抢沙发吧~