Volatile关键字与Java原子性的迷宫之旅
创始人
2024-11-14 11:05:34
0

在Java并发编程的领域中,有一种看似简单却蕴藏深意的关键字——volatile。它如同一把钥匙,打开了数据一致性与线程间通信的大门。但volatile并非全能,尤其是在原子性面前,它有着自己的局限性。本文将带你穿越迷宫,探索volatile的真谛及其与原子性的关系。

Volatile的面纱:可见性与有序性

Volatile关键字的核心作用在于确保变量的可见性有序性。所谓可见性,指的是当一个线程修改了某个volatile变量后,其他线程能够立即看到这一变化。有序性则保证了指令不会被处理器或编译器重排序,从而避免了某些类型的竞争条件。

示例代码:

public class VolatileVisibility {     public static volatile boolean ready = false;     public static int number = 0;      public static void main(String[] args) throws InterruptedException {         Thread writerThread = new Thread(() -> {             number = 42;             ready = true;         });          Thread readerThread = new Thread(() -> {             while (!ready) {                 Thread.yield();             }             System.out.println(number); // 应该输出42         });          writerThread.start();         readerThread.start();          writerThread.join();         readerThread.join();     } } 

在这个例子中,如果没有volatile修饰符,number的更新可能不会被readerThread及时发现,导致输出错误的结果。但加上volatile后,readerThread将能够正确读取到writerThread写入的新值。

原子性的迷雾:Volatile的盲区

虽然volatile能提供可见性和有序性,但它并不能保证原子性。原子性意味着一个操作要么完全执行,要么完全不执行,不会被中断。例如,考虑下面的代码片段:

示例代码:

public class VolatileNonAtomicity {     public static volatile int count = 0;      public static void main(String[] args) throws InterruptedException {         Thread[] threads = new Thread[10];         for (int i = 0; i < threads.length; i++) {             threads[i] = new Thread(() -> {                 for (int j = 0; j < 1000; j++) {                     count++; // 这个操作不是原子的                 }             });             threads[i].start();         }          for (Thread t : threads) {             t.join();         }          System.out.println("Expected: 10000, Actual: " + count);     } } 

尽管countvolatile的,但count++操作并非原子的,因此多个线程同时进行加法操作时,可能会得到比预期少的结果。这是因为count++实际上是由读取、计算、写回三个步骤组成的,而在多线程环境下,这些步骤可能被不同的线程交错执行。

原子性的解决方案:Java并发类库

为了弥补volatile在原子性方面的不足,Java提供了java.util.concurrent.atomic包,其中包含了一系列原子类,如AtomicIntegerAtomicLong等,它们利用内部的锁或其他同步机制来保证原子性。

示例代码:

import java.util.concurrent.atomic.AtomicInteger;  public class AtomicSafety {     public static AtomicInteger atomicCount = new AtomicInteger(0);      public static void main(String[] args) throws InterruptedException {         Thread[] threads = new Thread[10];         for (int i = 0; i < threads.length; i++) {             threads[i] = new Thread(() -> {                 for (int j = 0; j < 1000; j++) {                     atomicCount.incrementAndGet(); // 原子性操作                 }             });             threads[i].start();         }          for (Thread t : threads) {             t.join();         }          System.out.println("Expected: 10000, Actual: " + atomicCount.get());     } } 

在这个版本中,atomicCount使用AtomicInteger,其incrementAndGet()方法是原子性的,从而保证了所有线程操作的完整性,输出结果将始终等于预期。

相关内容

热门资讯

透视妙招!wepoker有没有... 透视妙招!wepoker有没有挂,wepoker亲友圈有用吗(脚本)透视教程(确实有挂)-哔哩哔哩1...
透视辅助!wpk透视(透视)w... 透视辅助!wpk透视(透视)wpk私人辅助,教程攻略(了解有挂)-哔哩哔哩1、进入游戏-大厅左侧-新...
有消息称!aapoker辅助怎... 有消息称!aapoker辅助怎么用,aapoker怎么选牌(透视)大纲教程(其实是真的挂)-哔哩哔哩...
透视演示!wepokerplu... 您好,wepoker正确养号方法这款游戏可以开挂的,确实是有挂的,需要了解加去威信【48527505...
透视透视!wpk有透视吗(透视... 透视透视!wpk有透视吗(透视)wpk可以透视吗,教程方式(真的有挂)-哔哩哔哩进入游戏-大厅左侧-...
现就发布提示!aapoker万... 现就发布提示!aapoker万能辅助器,aapoker万能辅助器(透视)策略教程(切实是有挂)-哔哩...
透视经验!wepokerh5破... 透视经验!wepokerh5破解,wepoker透视有没有(脚本)专业教程(一直真的有挂)-哔哩哔哩...
透视揭幕!wpk插件(透视)w... 透视揭幕!wpk插件(透视)wpk是真的还是假的,教程练习(有挂详情)-哔哩哔哩wpk是真的还是假的...
透视方针!wepoker辅助器... 透视方针!wepoker辅助器下载,wepoker软件靠谱么(脚本)科普教程(好像真的是有挂)-哔哩...
近年来!aapoker辅助器是... 近年来!aapoker辅助器是真的吗,aapoker公共底牌(透视)方案教程(原来有挂)-哔哩哔哩1...