JDK中多线程之原子类的JDK源码解读配合大神的一起看,秒懂。
来源:互联网 发布:剑灵龙女九头身数据图 编辑:程序博客网 时间:2024/06/05 11:40
一、 “JUC原子类”01之框架
1) 概要
根据修改的数据类型,可以将JUC包中的原子操作类可以分为4类。
1. 基本类型: AtomicInteger, AtomicLong, AtomicBoolean;
2. 数组类型: AtomicIntegerArray, AtomicLongArray,AtomicReferenceArray ;
3. 引用类型: AtomicReference, AtomicStampedRerence,AtomicMarkableReference ;
4. 对象的属性修改类型: AtomicIntegerFieldUpdater,AtomicLongFieldUpdater, AtomicReferenceFieldUpdater 。
这些类存在的目的是对相应的数据进行原子操作。所谓原子操作,是指操作过程不会被中断,保证数据操作是以原子方式进行的。
二、 “JUC原子类”02之 AtomicLong原子类
1) 概要
AtomicInteger,AtomicLong和AtomicBoolean这3个基本类型的原子类的原理和用法相似。本章以AtomicLong对基本类型的原子类进行介绍。内容包括:
AtomicLong介绍和函数列表
AtomicLong源码分析(基于JDK1.7.0_40)
AtomicLong示例
2) AtomicLong介绍和函数列表
AtomicLong是作用是对长整形进行原子操作。
在32位操作系统中,64位的long 和 double 变量由于会被JVM当作两个分离的32位来进行操作,所以不具有原子性。而使用AtomicLong能让long的操作保持原子型。
AtomicLong函数列表
// 构造函数
AtomicLong()
// 创建值为initialValue的AtomicLong对象
AtomicLong(longinitialValue)
// 以原子方式设置当前值为newValue。
final voidset(long newValue)
// 获取当前值
final long get()
// 以原子方式将当前值减 1,并返回减1后的值。等价于“--num”
final longdecrementAndGet()
// 以原子方式将当前值减 1,并返回减1前的值。等价于“num--”
final long getAndDecrement()
// 以原子方式将当前值加 1,并返回加1后的值。等价于“++num”
final longincrementAndGet()
// 以原子方式将当前值加 1,并返回加1前的值。等价于“num++”
final longgetAndIncrement()
// 以原子方式将delta与当前值相加,并返回相加后的值。
final longaddAndGet(long delta)
// 以原子方式将delta添加到当前值,并返回相加前的值。
final long getAndAdd(longdelta)
// 如果当前值 == expect,则以原子方式将该值设置为update。成功返回true,否则返回false,并且不修改原值。
final booleancompareAndSet(long expect, long update)
// 以原子方式设置当前值为newValue,并返回旧值。
final longgetAndSet(long newValue)
// 返回当前值对应的int值
int intValue()
// 获取当前值对应的long值
longlongValue()
// 以 float 形式返回当前值
floatfloatValue()
// 以 double 形式返回当前值
doubledoubleValue()
// 最后设置为给定值。延时设置变量值,这个等价于set()方法,但是由于字段是volatile类型的,因此次字段的修改会比普通字段(非volatile字段)有稍微的性能延时(尽管可以忽略),所以如果不是想立即读取设置的新值,允许在“后台”修改值,那么此方法就很有用。如果还是难以理解,这里就类似于启动一个后台线程如执行修改新值的任务,原线程就不等待修改结果立即返回(这种解释其实是不正确的,但是可以这么理解)。
final voidlazySet(long newValue)
// 如果当前值 == 预期值,则以原子方式将该设置为给定的更新值。JSR规范中说:以原子方式读取和有条件地写入变量但不 创建任何happen-before 排序,因此不提供与除 weakCompareAndSet 目标外任何变量以前或后续读取或写入操作有关的任何保证。大意就是说调用weakCompareAndSet时并不能保证不存在happen-before的发生(也就是可能存在指令重排序导致此操作失败)。但是从Java源码来看,其实此方法并没有实现JSR规范的要求,最后效果和compareAndSet是等效的,都调用了unsafe.compareAndSwapInt()完成操作。
final booleanweakCompareAndSet(long expect, long update)
3) AtomicLong源码分析(基于JDK1.7.0_40)
AtomicLong的代码很简单,下面仅以incrementAndGet()为例,对AtomicLong的原理进行说明。
incrementAndGet()源码如下:
public final longincrementAndGet() {
for (;;) {
// 获取AtomicLong当前对应的long值
long current = get();
// 将current加1
long next = current + 1;
// 通过CAS函数,更新current的值
if (compareAndSet(current, next))
return next;
}
}
说明:
(01)incrementAndGet()首先会根据get()获取AtomicLong对应的long值。该值是volatile类型的变量,get()的源码如下:
// value是AtomicLong对应的long值
private volatilelong value;
// 返回AtomicLong对应的long值
public final longget() {
return value;
}
(02)incrementAndGet()接着将current加1,然后通过CAS函数,将新的值赋值给value。
compareAndSet()的源码如下:
public finalboolean compareAndSet(long expect, long update) {
return unsafe.compareAndSwapLong(this,valueOffset, expect, update);
}
compareAndSet()的作用是更新AtomicLong对应的long值。它会比较AtomicLong的原始值是否与expect相等,若相等的话,则设置AtomicLong的值为update。
4) AtomicLong示例
// LongTest.java的源码
importjava.util.concurrent.atomic.AtomicLong;
public classLongTest {
public static void main(String[] args){
// 新建AtomicLong对象
AtomicLong mAtoLong = new AtomicLong();
mAtoLong.set(0x0123456789ABCDEFL);
System.out.printf("%20s :0x%016X\n", "get()", mAtoLong.get());
System.out.printf("%20s :0x%016X\n", "intValue()", mAtoLong.intValue());
System.out.printf("%20s :0x%016X\n", "longValue()", mAtoLong.longValue());
System.out.printf("%20s :%s\n", "doubleValue()", mAtoLong.doubleValue());
System.out.printf("%20s :%s\n", "floatValue()", mAtoLong.floatValue());
System.out.printf("%20s :0x%016X\n", "getAndDecrement()", mAtoLong.getAndDecrement());
System.out.printf("%20s :0x%016X\n", "decrementAndGet()", mAtoLong.decrementAndGet());
System.out.printf("%20s :0x%016X\n", "getAndIncrement()", mAtoLong.getAndIncrement());
System.out.printf("%20s :0x%016X\n", "incrementAndGet()", mAtoLong.incrementAndGet());
System.out.printf("%20s :0x%016X\n", "addAndGet(0x10)", mAtoLong.addAndGet(0x10));
System.out.printf("%20s :0x%016X\n", "getAndAdd(0x10)", mAtoLong.getAndAdd(0x10));
System.out.printf("\n%20s :0x%016X\n", "get()", mAtoLong.get());
System.out.printf("%20s :%s\n", "compareAndSet()", mAtoLong.compareAndSet(0x12345679L,0xFEDCBA9876543210L));
System.out.printf("%20s :0x%016X\n", "get()", mAtoLong.get());
}
}
运行结果:
get() : 0x0123456789ABCDEF
intValue() : 0x0000000089ABCDEF
longValue() : 0x0123456789ABCDEF
doubleValue() : 8.1985529216486896E16
floatValue() : 8.1985531E16
getAndDecrement() : 0x0123456789ABCDEF
decrementAndGet() : 0x0123456789ABCDED
getAndIncrement() : 0x0123456789ABCDED
incrementAndGet() : 0x0123456789ABCDEF
addAndGet(0x10) : 0x0123456789ABCDFF
getAndAdd(0x10) : 0x0123456789ABCDFF
get() : 0x0123456789ABCE0F
compareAndSet() : false
get() : 0x0123456789ABCE0F
三、 “JUC原子类”03之 AtomicLongArray原子类
1) 概要
AtomicIntegerArray,AtomicLongArray, AtomicReferenceArray这3个数组类型的原子类的原理和用法相似。本章以AtomicLongArray对数组类型的原子类进行介绍。内容包括:
AtomicLongArray介绍和函数列表
AtomicLongArray源码分析(基于JDK1.7.0_40)
AtomicLongArray示例
2) AtomicLongArray介绍和函数列表
在AtomicLong原子类"中介绍过,AtomicLong是作用是对长整形进行原子操作。而AtomicLongArray的作用则是对"长整形数组"进行原子操作。
AtomicLongArray函数列表
// 创建给定长度的新 AtomicLongArray。
AtomicLongArray(intlength)
// 创建与给定数组具有相同长度的新 AtomicLongArray,并从给定数组复制其所有元素。
AtomicLongArray(long[]array)
// 以原子方式将给定值添加到索引 i 的元素。
long addAndGet(inti, long delta)
// 如果当前值 == 预期值,则以原子方式将该值设置为给定的更新值。
booleancompareAndSet(int i, long expect, long update)
// 以原子方式将索引 i 的元素减1。
longdecrementAndGet(int i)
// 获取位置 i 的当前值。
long get(int i)
// 以原子方式将给定值与索引 i 的元素相加。
long getAndAdd(inti, long delta)
// 以原子方式将索引 i 的元素减 1。
longgetAndDecrement(int i)
// 以原子方式将索引 i 的元素加 1。
longgetAndIncrement(int i)
// 以原子方式将位置 i 的元素设置为给定值,并返回旧值。
long getAndSet(inti, long newValue)
// 以原子方式将索引 i 的元素加1。
longincrementAndGet(int i)
// 最终将位置 i 的元素设置为给定值。
void lazySet(inti, long newValue)
// 返回该数组的长度。
int length()
// 将位置 i 的元素设置为给定值。
void set(int i,long newValue)
// 返回数组当前值的字符串表示形式。
String toString()
// 如果当前值 == 预期值,则以原子方式将该值设置为给定的更新值。
boolean weakCompareAndSet(int i, long expect, longupdate)
3) AtomicLongArray源码分析(基于JDK1.7.0_40)
AtomicLongArray的代码很简单,下面仅以incrementAndGet()为例,对AtomicLong的原理进行说明。
incrementAndGet()源码如下:
public final longincrementAndGet(int i) {
return addAndGet(i, 1);
}
说明:incrementAndGet()的作用是以原子方式将long数组的索引 i 的元素加1,并返回加1之后的值。
addAndGet()源码如下:
public longaddAndGet(int i, long delta) {
// 检查数组是否越界
long offset = checkedByteOffset(i);
while (true) {
// 获取long型数组的索引 offset 的原始值
long current = getRaw(offset);
// 修改long型值
long next = current + delta;
// 通过CAS更新long型数组的索引offset的值。
if (compareAndSetRaw(offset, current,next))
return next;
}
}
说明:addAndGet()首先检查数组是否越界。如果没有越界的话,则先获取数组索引i的值;然后通过CAS函数更新i的值。
getRaw()源码如下:
private longgetRaw(long offset) {
return unsafe.getLongVolatile(array,offset);
}
说明:unsafe是通过Unsafe.getUnsafe()返回的一个Unsafe对象。通过Unsafe的CAS函数对long型数组的元素进行原子操作。如compareAndSetRaw()就是调用Unsafe的CAS函数,它的源码如下:
private booleancompareAndSetRaw(long offset, long expect, long update) {
return unsafe.compareAndSwapLong(array,offset, expect, update);
}
4) AtomicLongArray示例
//LongArrayTest.java的源码
importjava.util.concurrent.atomic.AtomicLongArray;
public classLongArrayTest {
public static void main(String[] args){
// 新建AtomicLongArray对象
long[] arrLong = new long[] {10, 20,30, 40, 50};
AtomicLongArray ala = newAtomicLongArray(arrLong);
ala.set(0, 100);
for (int i=0, len=ala.length();i<len; i++)
System.out.printf("get(%d) :%s\n", i, ala.get(i));
System.out.printf("%20s :%s\n", "getAndDecrement(0)", ala.getAndDecrement(0));
System.out.printf("%20s :%s\n", "decrementAndGet(1)", ala.decrementAndGet(1));
System.out.printf("%20s :%s\n", "getAndIncrement(2)", ala.getAndIncrement(2));
System.out.printf("%20s :%s\n", "incrementAndGet(3)", ala.incrementAndGet(3));
System.out.printf("%20s :%s\n", "addAndGet(100)", ala.addAndGet(0, 100));
System.out.printf("%20s :%s\n", "getAndAdd(100)", ala.getAndAdd(1, 100));
System.out.printf("%20s :%s\n", "compareAndSet()", ala.compareAndSet(2, 31, 1000));
System.out.printf("%20s :%s\n", "get(2)", ala.get(2));
}
}
运行结果:
get(0) : 100
get(1) : 20
get(2) : 30
get(3) : 40
get(4) : 50
getAndDecrement(0) : 100
decrementAndGet(1) : 19
getAndIncrement(2) : 30
incrementAndGet(3) : 41
addAndGet(100) : 199
getAndAdd(100) : 19
compareAndSet() : true
get(2) : 1000
四、 “JUC原子类”04之 AtomicReference原子类
1) 概要
本章对AtomicReference引用类型的原子类进行介绍。内容包括:
AtomicReference介绍和函数列表
AtomicReference源码分析(基于JDK1.7.0_40)
AtomicReference示例
2) AtomicReference介绍和函数列表
AtomicReference是作用是对"对象"进行原子操作。
AtomicReference函数列表
// 使用 null 初始值创建新的AtomicReference。
AtomicReference()
// 使用给定的初始值创建新的 AtomicReference。
AtomicReference(VinitialValue)
// 如果当前值 == 预期值,则以原子方式将该值设置为给定的更新值。
booleancompareAndSet(V expect, V update)
// 获取当前值。
V get()
// 以原子方式设置为给定值,并返回旧值。
V getAndSet(VnewValue)
// 最终设置为给定值。
void lazySet(VnewValue)
// 设置为给定值。
void set(VnewValue)
// 返回当前值的字符串表示形式。
String toString()
// 如果当前值 == 预期值,则以原子方式将该值设置为给定的更新值。
booleanweakCompareAndSet(V expect, V update)
3) AtomicReference源码分析(基于JDK1.7.0_40)
在JDK1.7.0_40中AtomicReference.java的源码如下:
public classAtomicReference<V> implementsjava.io.Serializable {
private static final long serialVersionUID= -1848883965231344442L;
// 获取Unsafe对象,Unsafe的作用是提供CAS操作
private static final Unsafe unsafe =Unsafe.getUnsafe();
private static final long valueOffset;
static {
try {
valueOffset = unsafe.objectFieldOffset
(AtomicReference.class.getDeclaredField("value"));
} catch (Exception ex) { throw newError(ex); }
}
// volatile类型
private volatile V value;
public AtomicReference(V initialValue) {
value = initialValue;
}
public AtomicReference() {
}
public final V get() {
return value;
}
public final void set(V newValue) {
value = newValue;
}
public final void lazySet(V newValue) {
unsafe.putOrderedObject(this,valueOffset, newValue);
}
public final boolean compareAndSet(Vexpect, V update) {
returnunsafe.compareAndSwapObject(this, valueOffset, expect, update);
}
public final boolean weakCompareAndSet(Vexpect, V update) {
returnunsafe.compareAndSwapObject(this, valueOffset, expect, update);
}
public final V getAndSet(V newValue) {
while (true) {
V x = get();
if (compareAndSet(x, newValue))
return x;
}
}
public String toString() {
return String.valueOf(get());
}
}
说明:
AtomicReference的源码比较简单。它是通过"volatile"和"Unsafe提供的CAS函数实现"原子操作。
(01) value是volatile类型。这保证了:当某线程修改value的值时,其他线程看到的value值都是最新的value值,即修改之后的volatile的值。
(02) 通过CAS设置value。这保证了:当某线程池通过CAS函数(如compareAndSet函数)设置value时,它的操作是原子的,即线程在操作value时不会被中断。
4) AtomicReference示例
AtomicReference示例
//AtomicReferenceTest.java的源码
importjava.util.concurrent.atomic.AtomicReference;
public classAtomicReferenceTest {
public static void main(String[] args){
// 创建两个Person对象,它们的id分别是101和102。
Person p1 = new Person(101);
Person p2 = new Person(102);
// 新建AtomicReference对象,初始化它的值为p1对象
AtomicReference ar = newAtomicReference(p1);
// 通过CAS设置ar。如果ar的值为p1的话,则将其设置为p2。
ar.compareAndSet(p1, p2);
Person p3 = (Person)ar.get();
System.out.println("p3 is"+p3);
System.out.println("p3.equals(p1)="+p3.equals(p1));
}
}
class Person {
volatile long id;
public Person(long id) {
this.id = id;
}
public String toString() {
return "id:"+id;
}
}
运行结果:
p3 is id:102
p3.equals(p1)=false
结果说明:
新建AtomicReference对象ar时,将它初始化为p1。
紧接着,通过CAS函数对它进行设置。如果ar的值为p1的话,则将其设置为p2。
最后,获取ar对应的对象,并打印结果。p3.equals(p1)的结果为false,这是因为Person并没有覆盖equals()方法,而是采用继承自Object.java的equals()方法;而Object.java中的equals()实际上是调用"=="去比较两个对象,即比较两个对象的地址是否相等。
五、 “JUC原子类”05之 AtomicLongFieldUpdater原子类
1) 概要
AtomicIntegerFieldUpdater,AtomicLongFieldUpdater和AtomicReferenceFieldUpdater这3个修改类的成员的原子类型的原理和用法相似。本章以对基本类型的原子类进行介绍。内容包括:
AtomicLongFieldUpdater介绍和函数列表
AtomicLongFieldUpdater示例
AtomicLongFieldUpdater源码分析(基于JDK1.7.0_40)
2) AtomicLongFieldUpdater介绍和函数列表
AtomicLongFieldUpdater可以对指定"类的 'volatilelong'类型的成员"进行原子更新。它是基于反射原理实现的。
AtomicLongFieldUpdater函数列表
// 受保护的无操作构造方法,供子类使用。
protectedAtomicLongFieldUpdater()
// 以原子方式将给定值添加到此更新器管理的给定对象的字段的当前值。
long addAndGet(Tobj, long delta)
// 如果当前值 == 预期值,则以原子方式将此更新器所管理的给定对象的字段设置为给定的更新值。
abstract booleancompareAndSet(T obj, long expect, long update)
// 以原子方式将此更新器管理的给定对象字段当前值减 1。
longdecrementAndGet(T obj)
// 获取此更新器管理的在给定对象的字段中保持的当前值。
abstract longget(T obj)
// 以原子方式将给定值添加到此更新器管理的给定对象的字段的当前值。
long getAndAdd(Tobj, long delta)
// 以原子方式将此更新器管理的给定对象字段当前值减 1。
longgetAndDecrement(T obj)
// 以原子方式将此更新器管理的给定对象字段的当前值加 1。
longgetAndIncrement(T obj)
// 将此更新器管理的给定对象的字段以原子方式设置为给定值,并返回旧值。
long getAndSet(Tobj, long newValue)
// 以原子方式将此更新器管理的给定对象字段当前值加 1。
longincrementAndGet(T obj)
// 最后将此更新器管理的给定对象的字段设置为给定更新值。
abstract voidlazySet(T obj, long newValue)
// 为对象创建并返回一个具有给定字段的更新器。
static <U>AtomicLongFieldUpdater<U> newUpdater(Class<U> tclass, StringfieldName)
// 将此更新器管理的给定对象的字段设置为给定更新值。
abstract voidset(T obj, long newValue)
// 如果当前值 == 预期值,则以原子方式将此更新器所管理的给定对象的字段设置为给定的更新值。
abstract booleanweakCompareAndSet(T obj, long expect, long update)
3) AtomicLongFieldUpdater示例
// LongTest.java的源码
importjava.util.concurrent.atomic.AtomicLongFieldUpdater;
public classLongFieldTest {
public static void main(String[] args) {
// 获取Person的class对象
Class cls = Person.class;
// 新建AtomicLongFieldUpdater对象,传递参数是“class对象”和“long类型在类中对应的名称”
AtomicLongFieldUpdater mAtoLong =AtomicLongFieldUpdater.newUpdater(cls, "id");
Person person = new Person(12345678L);
// 比较person的"id"属性,如果id的值为12345678L,则设置为1000。
mAtoLong.compareAndSet(person,12345678L, 1000);
System.out.println("id="+person.getId());
}
}
class Person {
volatile long id;
public Person(long id) {
this.id = id;
}
public void setId(long id) {
this.id = id;
}
public long getId() {
return id;
}
}
运行结果:
id=1000
4) AtomicLongFieldUpdater源码分析(基于JDK1.7.0_40)
下面分析LongFieldTest.java的流程。
1. newUpdater()
newUpdater()的源码如下:
public static<U> AtomicLongFieldUpdater<U> newUpdater(Class<U> tclass,String fieldName) {
Class<?> caller =Reflection.getCallerClass();
if (AtomicLong.VM_SUPPORTS_LONG_CAS)
return new CASUpdater<U>(tclass,fieldName, caller);
else
return newLockedUpdater<U>(tclass, fieldName, caller);
}
说明:newUpdater()的作用是获取一个AtomicIntegerFieldUpdater类型的对象。
它实际上返回的是CASUpdater对象,或者LockedUpdater对象;具体返回哪一个类取决于JVM是否支持long类型的CAS函数。CASUpdater和LockedUpdater都是AtomicIntegerFieldUpdater的子类,它们的实现类似。下面以CASUpdater来进行说明。
CASUpdater类的源码如下:
public booleancompareAndSet(T obj, long expect, long update) {
if (obj == null || obj.getClass() != tclass|| cclass != null) fullCheck(obj);
return unsafe.compareAndSwapLong(obj,offset, expect, update);
}
说明:它实际上是通过CAS函数操作。如果类的long对象的值是expect,则设置它的值为update。
大神总结的目录:http://www.cnblogs.com/skywang12345/p/3323085.html(转载),仅供个人学习,如有抄袭请包容(我也忘了cry....)
- JDK中多线程之原子类的JDK源码解读配合大神的一起看,秒懂。
- JDK中多线程之JUC锁的JDK源码解读配合大神的一起看,秒懂。
- JDK中多线程之JUC集合的JDK源码解读配合大神的一起看,秒懂。
- JDK中多线程之JUC线程池的JDK源码解读配合大神的一起看,秒懂。
- JDK中IO包的源码解读配合大神的一起看,秒懂。
- JDK中集合包MAP的源码解读配合大神的一起看,秒懂。
- JDK中集合包Collection和List的源码解读配合大神的一起看,秒懂。
- JDK中Future的源码解读
- JDK的HashMap源码解读
- HashSet的故事----Jdk源码解读
- JDK源码解读之RegularEnumSet
- JDK之ArrayList源码解读
- JDK之CopyOnWriteArrayList源码解读
- JDK之HashMap源码解读
- JDK中ExecutorService源码解读
- jdk原生态的WebService
- 原子类与多线程中变量的累加问题
- java源码解读之ArrayList------jdk 1.7
- UNIX网络编程卷一 第四章 基本TCP套接字编程
- java多线程通信
- mysql 索引与优化 之 索引算法
- FZU2216【二分】
- Qt:QML:文字自动滚动
- JDK中多线程之原子类的JDK源码解读配合大神的一起看,秒懂。
- 从C#到TypeScrip
- hdu 4513左边到中间递增的回文串
- HDU 1081 To The Max
- html转换成excel表格导出兼容ie,火狐,谷歌
- 信息安全第三节课总结
- 【Android】Glide 如何获取 bitmap 来进行再处理(比如处理长图的显示)
- 《百万美元宝贝》
- css-5