AtomicLong源码解析

一、前言

    AtomicLong是作用是对长整形进行原子操作,是java.util.concurrent.atomic包下的一个提供原子操作的Long类型数据的类。在32位操作系统中,64位的long 和 double 变量由于会被JVM当作两个分离的32位来进行操作,所以不具有原子性。而使用AtomicLong能让long的操作保持原子型。

二、源码解析

继承了哪些类

在这里插入图片描述
    由上图可知AtomicLong继承了Number类和Serializable接口,Serializable接口我想你们应该知道了其作用,但是Number是干什么可能还没有了解。一句话来说明Number类:所有的包装类(Integer、Long、Byte、Double、Float、Short)都是抽象类Number的子类

变量说明

//序列化序号,用来对对象进行反序列化时的判断
private static final long serialVersionUID = 1927816293512124184L;
//unsafe常量,设置为使用Unsafe.compareAndSwapInt进行更新
private static final Unsafe unsafe = Unsafe.getUnsafe();
//AtomicLong的值在内存地址的偏移量
private static final long valueOffset;
// 记录底层JVM是否支持无锁比较和交换long。
// 虽然Unsafe.compareAndSwapLong方法在这两种情况下都可以工作
// 但是一些结构应该在Java级别进行处理,以避免锁定用户可见的锁。
static final boolean VM_SUPPORTS_LONG_CAS = VMSupportsCS8();
//AtomicLong当前值
private volatile long value;

valueOffset是如何获取的?

static {
    try {
        //获取valueOffset的偏移量
        valueOffset = unsafe.objectFieldOffset
            (AtomicLong.class.getDeclaredField("value"));
    } catch (Exception ex) { throw new Error(ex); }
}

    由源码得知是利用Unsafe类的内部方法进行对“value”获取其内存的偏移地址的,利用static的特性在类加载到内存中就获取到valueOffset的值。

构造方法

AtomicLong类的构造方法只有两个,如下所示:

//使用给定的初始值创建一个新的AtomicLong,即设置当前AtomicLong的值
public AtomicLong(long initialValue) {
    value = initialValue;
}
//第二个构造方法是默认的构造方法
public AtomicLong() {
}

其他方法

① get()方法。

//直接返回变量value
public final long get() {
    return value;
}

② set(long newValue)方法。

//通过参数newValue将变量value进行值的更新
public final void set(long newValue) {
    value = newValue;
}

③ lazySet(long newValue) 方法。

//通过unsafe变量最终设置为给定的值。
public final void lazySet(long newValue) {
    unsafe.putOrderedLong(this, valueOffset, newValue);
}

④ getAndSet(long newValue)方法。

//先获取旧值再更新新值,还是利用unsafe的内部方法来进行操作
public final long getAndSet(long newValue) {
    return unsafe.getAndSetLong(this, valueOffset, newValue);
}

⑤ compareAndSet(long expect, long update)方法。

//CAS方法,利用判断旧值符合预期值并且更新新的值
public final boolean compareAndSet(long expect, long update) {
    return unsafe.compareAndSwapLong(this, valueOffset, expect, update);
}

⑥ getAndIncrement()方法。

//先获取旧值然后再旧值上加一,利用的unsafe内部方法
public final long getAndIncrement() {
    return unsafe.getAndAddLong(this, valueOffset, 1L);
}

⑦ getAndAdd(long delta)方法。

//先获取旧值然后再旧值上减一,利用的unsafe内部方法
public final long getAndDecrement() {
    return unsafe.getAndAddLong(this, valueOffset, -1L);
}

     当然还有一些其他方法没有写出来,不过其内容都是利用unsafe变量的内部的本地方法进行操作,大家可以自己去了解。

     备注:java.util.concurrent.atomic包下是没有AtomicFloat和AtomicDouble这两个方法的,也就是说没有对float和double的原子类操作,因为这两个操作在AtomicLong和AtomicInteger都有两个方法floatValue()和doubleValue(),其源码如下:

//返回float类型的值,利用强制转化
public float floatValue() {
    return (float)get();
}
//返回double类型的值,利用强制转化
public double doubleValue() {
    return (double)get();
}

三、总结

  1. AtomicLong中的内部方法的实现都是利用他的Unsafe变量来进行操作的。
  2. 大部分的方法都是final的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值