atomic包之AtomicBoolean

本文详细介绍了Java并发包java.util.concurrent下的AtomicBoolean类,包括其源码解析、如何通过CAS操作实现线程安全,以及volatile关键字的作用机制。

atomic简介

atomic意思是原子;atomic是java.util.concurrent下的专门为线程安全设计的Java包,包含多个原子操作类,主要通过CAS(Compare And Set)来确保多线程下操作的安全性;

AtomicBoolean

  • 源码

    public class AtomicBoolean implements java.io.Serializable {
    private static final long serialVersionUID = 4654671469794556979L;
    private static final Unsafe unsafe = Unsafe.getUnsafe();
    private static final long valueOffset;
    static {
        try {
            // 通过unsafe获取value的内存地址
            valueOffset = unsafe.objectFieldOffset
                (AtomicBoolean.class.getDeclaredField("value"));
        } catch (Exception ex) { throw new Error(ex); }
    }
    // value 被定义为volatile(内存可见)
    private volatile int value;
    // 构造方法:true用1表示
    public AtomicBoolean(boolean initialValue) {
        value = initialValue ? 1 : 0;
    }
    public AtomicBoolean() {
    }
    public final boolean get() {
        // 默认为false
        return value != 0;
    }
    public final boolean compareAndSet(boolean expect, boolean update) {
        int e = expect ? 1 : 0;
        int u = update ? 1 : 0;
        //通过调用unsafe的native方法来实现CAS操作,jdk的注释是:
        //如果valueOffset的内存地址的存储的值为e,那么将值更新为u;
        //该native方法通过调用硬件来搞定,而value又定义为volatile,就可以实现线程安全
        return unsafe.compareAndSwapInt(this, valueOffset, e, u);
    }
    public boolean weakCompareAndSet(boolean expect, boolean update) {
        int e = expect ? 1 : 0;
        int u = update ? 1 : 0;
        return unsafe.compareAndSwapInt(this, valueOffset, e, u);
    }
    public final void set(boolean newValue) {
        value = newValue ? 1 : 0;
    }
    public final void lazySet(boolean newValue) {
        int v = newValue ? 1 : 0;
        unsafe.putOrderedInt(this, valueOffset, v);
    }
    public final boolean getAndSet(boolean newValue) {
        boolean prev;
        do {
            prev = get();
        } while (!compareAndSet(prev, newValue));
        return prev;
    }
    public String toString() {
        return Boolean.toString(get());
    }
    }

总结

atomic是通过cpu来进行硬件上的阻塞及volatile来实现原子操作的线程安全的;所以在java语言层面是非阻塞的,但是在底层硬件上是阻塞的;

在这里提一下volatile关键字,用volatile定义的变量是内存可见的,其实现也是通过硬件来实现的;处理器识别到从内存中读取的数据是可缓存的,则处理器读取整个缓存行到缓存,下次访问的时候从缓存中读取,当用volatile修饰的变量有了修改,会生成一条lock前缀的指令,lock指令做了两件事情:

  • 将当前处理器的缓存行的数据刷到内存中;
  • 使其他cpu缓存了这个地址的数据无效,重新从内存读取,这样就实现了volatile修饰的变量内存可见
Java 的 `atomic` 提供了一系列原子操作类,这些类可以便捷、高效、安全地使用在需要更新变量的场景。涵盖的操作类有原子更新基本类型、原子更新数组、原子更新引用、原子更新属性,且都是使用 `Unsafe` 类提供的三个 CAS 方法结合死循环实现,即乐观锁[^1]。以下是常见的 `Atomic` 类及用法: ### 基本类型原子类 - **`AtomicInteger`**:用于原子更新 `int` 类型的值。示例代码如下: ```java import java.util.concurrent.atomic.AtomicInteger; public class AtomicDemo { private static AtomicInteger atomicInteger = new AtomicInteger(1); public static void main(String[] args) { System.out.println(atomicInteger.getAndIncrement()); System.out.println(atomicInteger.get()); } } ``` 上述代码会输出: ``` 1 2 ``` - **`AtomicLong`**:原子更新 `long` 类型的值。 - **`AtomicBoolean`**:原子更新 `boolean` 类型的值。示例代码如下: ```java import java.util.concurrent.atomic.AtomicBoolean; public class CurrentPackage { public static void main(String[] args) throws InterruptedException { AtomicBoolean atomicBoolean = new AtomicBoolean(true); Thread thread = new Thread(() -> { while (atomicBoolean.get()) { System.out.println("hello"); } }); Thread th = new Thread(() -> { atomicBoolean.compareAndExchange(true, false); }); thread.start(); Thread.sleep(1); th.start(); } } ``` ### 引用类型原子类 - **`AtomicReference<V>`**:原子更新对象引用。 - **`AtomicStampedReference<V>`**:可解决 ABA 问题。 - **`AtomicMarkableReference<V>`**:标记对象是否被修改。 ### 数组类型原子类 - **`AtomicIntegerArray`**:原子更新 `int` 数组。 - **`AtomicLongArray`**:原子更新 `long` 数组。 - **`AtomicReferenceArray`**:原子更新对象引用数组。 ### 字段更新器(FieldUpdaters) - **`AtomicIntegerFieldUpdater`**:更新对象的 `int` 字段。 ### 适用场景与局限性 `Atomic` 类适用于需要对变量进行原子更新的场景。不过,尽管 `AtomicInteger` 提供了原子性的保证,但它不能保证操作的“逻辑原子性”。例如,若需要检查值,然后基于这个值进行更新(如计数器小于 10 就增加),这样的操作序列无法保证原子性,因为该过程涉及多个步骤,而 CAS 只能保证单个操作的原子性[^5]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值