一、前言
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();
}
三、总结
- AtomicLong中的内部方法的实现都是利用他的Unsafe变量来进行操作的。
- 大部分的方法都是final的。