• 你对java的原子性了解多少?


    你对java的原子性了解多少?

    java里的原子性是什么

    在Java中,原子性是指一个操作是不可被中断的整体操作。原子性确保一个操作在多线程环境下执行时,不会被其他线程干扰,要么完全执行成功,要么完全不执行。

    Java提供了多种机制来实现原子性操作:

    1. volatile关键字:使用volatile修饰的变量可以保证变量的可见性,并且对该变量的读写操作都是原子的。它可以用于简单类型的变量(如int、boolean等),但对于复合操作(如i++)则不能保证原子性。
    2. synchronized关键字:synchronized可以用来实现代码块或方法的同步,可以保证同一时刻只有一个线程访问被锁定的代码块或方法。这样可以防止多个线程同时执行该代码块或方法,从而保证了操作的原子性。
    3. Atomic包:Java.util.concurrent.atomic包提供了一系列的原子类,如AtomicInteger、AtomicLong、AtomicBoolean等。这些类利用底层的CAS(Compare and Swap)操作实现了原子性的读取和修改。通过这些原子类,我们可以对变量进行原子操作,而无需显式地使用锁。

    java实现原子性的原理是什么

    Java实现原子性的原理主要依赖于底层硬件和虚拟机的支持。以下是一些关键的原理:

    1. 内存模型:Java内存模型(Java Memory Model,JMM)定义了多线程环境下共享变量的可见性、有序性和原子性规则。JMM确保了在不同线程之间对共享变量的读写操作的一致性。
    2. 原子操作指令:底层硬件提供了一些原子操作指令,如CAS(Compare and Swap)指令。CAS指令可以比较内存中的值与期望值,如果相等,则将新值写入内存;否则,重新比较并尝试更新值。这个过程是原子的,即不会被其他线程干扰。
    3. 内存屏障(Memory Barrier):内存屏障是一种硬件或软件机制,用于控制内存访问的顺序和可见性。它会强制执行一些顺序性和可见性的保证,以确保原子性操作的正确性。内存屏障可以防止指令重排序和缓存一致性问题。
    4. Java中的锁机制:Java提供了synchronized关键字和Lock接口来实现锁机制,用于保证关键代码块的互斥访问。通过获取锁,其他线程将被阻塞,从而实现操作的原子性。

    java如何实现原子性

    Java提供了几种方式来实现原子性操作:

    1. volatile 关键字:将一个变量声明为 volatile,可以保证对该变量的读写操作都具有原子性。volatile 可以保证对变量的写操作在多线程环境下的可见性,并防止指令重排序带来的问题。
    2. synchronized 关键字:使用 synchronized 关键字可以实现代码块或方法级别的同步,保证同一时刻只有一个线程执行被锁定的代码。通过获取对象的监视器锁(内置锁),可以确保被锁定的代码块或方法的执行是原子的。
    3. 原子类:Java.util.concurrent.atomic 包提供了一系列的原子类,如 AtomicIntegerAtomicLongAtomicBoolean 等。这些类利用底层的 CAS(Compare and Swap)操作实现了原子性的读取和修改。通过使用这些原子类,我们可以对变量进行原子操作,而无需显式地使用锁机制。
    4. 锁机制:Java 中的锁机制,如 Lock 接口和 ReentrantLock 类,可以实现更细粒度的原子性控制。使用锁可以保证代码块或方法在同一时刻只被一个线程访问,从而实现操作的原子性。

    原子类 AtomicInteger 实现原子性的例子:

    import java.util.concurrent.atomic.AtomicInteger;
    
    public class AtomicCounter {
        private AtomicInteger count = new AtomicInteger(0);
    
        public void increment() {
            count.incrementAndGet();
        }
    
        public int getCount() {
            return count.get();
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    我们使用了 AtomicInteger 类来存储计数器的值,并定义了 increment() 方法来增加计数器的值。incrementAndGet() 方法可以保证对计数器的操作具有原子性。

    以下是一个简单的测试程序,演示了多线程访问计数器时的效果:

    public class AtomicCounterTest {
        public static void main(String[] args) throws InterruptedException {
            final int threadCount = 10;
            final int targetCount = 10000;
            final AtomicCounter counter = new AtomicCounter();
    
            // 创建多个线程并启动
            Thread[] threads = new Thread[threadCount];
            for (int i = 0; i < threadCount; i++) {
                threads[i] = new Thread(() -> {
                    for (int j = 0; j < targetCount; j++) {
                        counter.increment();
                    }
                });
                threads[i].start();
            }
    
            // 等待所有线程执行完成
            for (Thread thread : threads) {
                thread.join();
            }
    
            // 输出最终结果
            System.out.println("Final Count: " + counter.getCount());
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    我们创建了多个线程,并启动它们来并发地访问计数器。每个线程都会调用 increment() 方法,将计数器的值增加指定的次数。最终,我们输出计数器的最终值。

    通过运行这个测试程序,可以看到无论多少个线程执行,最终计数器的值都是正确的,这得益于 AtomicInteger 类提供了原子性的操作。

  • 相关阅读:
    Ceres 自动求导解析-从原理到实践
    企业级API网关,金蝶是如何架构的?
    MyBatis拦截器优雅实现数据脱敏
    STM32CubeIDE、HAL、OLED、MPU6050学习笔记
    c语言分层理解(c语言字符串+内存库函数)
    PostgreSQL数据库统计信息——compute_scalar_stats计算统计数据
    LabVIEW中将前面板置于所有桌面窗口的前面
    springcloud-gateway 路由加载流程
    彻底搞懂MySQL事务
    groovy:调用jenkins任务时的请求失败问题
  • 原文地址:https://blog.csdn.net/weixin_44427181/article/details/132948607