今天在看AtomicInteger,看到自增方法时候有点迷糊,这里整理一下:
public final int getAndIncrement() {
return unsafe.getAndAddInt(this, valueOffset, 1);
}
public final int incrementAndGet() {
return unsafe.getAndAddInt(this, valueOffset, 1) + 1;
}
我第一次试验代码是这么写的:
AtomicInteger first = new AtomicInteger(5);
first.getAndIncrement();
System.out.println(first);
AtomicInteger second = new AtomicInteger(5);
second.getAndIncrement();
System.out.println(second);
输出结果为:
6
6
对这个结果,很是迷惑,看了半天也没发现问题。
后来看到这两个方法返回值是 int 类型才反应过来,不应该拿AtomicInteger对象直接输出,应该拿个变量存放一下。
还有,需要注意,
getAndAddInt 方法中返回的值,不是修改后 (var5+var4) 的值,而是 var5 !
public final int getAndAddInt(Object var1, long var2, int var4) {
int var5;
do {
var5 = this.getIntVolatile(var1, var2);
} while(!this.compareAndSwapInt(var1, var2, var5, var5 + var4));
return var5;
}
this.compareAndSwapInt(var1, var2, var5, var5 + var4)
public final native boolean compareAndSwapInt(this, valueOffset, expect, update) 方法有四个参数,其中第一个参数为需要改变的对象,第二个参数为偏移量, 第三个参数为期待的值,第四个参数为更新后的值。整个方法的作用即为若调用该方法时,value的值与expect这个值相等,那么则将value修改为update这个值,并返回一个true,如果调用该方法时,value的值与expect这个值不相等,那么不做任何操作,并返回一个false。
getAndAddInt 方法此处有个do while循环,可以
保证如果调用compareAndSwapInt
这个方法返回为false时,能再次尝试进行修改value的值,直到修改成功,并返回修改前value的值。
应该修改为这样子:
AtomicInteger first = new AtomicInteger(5);
int a = first.getAndIncrement();
System.out.println("a value:"+a);
System.out.println("first value:"+first);
AtomicInteger second = new AtomicInteger(5);
int b = second.incrementAndGet();
System.out.println("b value:"+b);
System.out.println("seconde value:"+second);
输出结果为:
a value: 5
first value: 6
b value: 6
seconde value: 6
这样子才对。
所以,这两个方法区别就很简单了,一个是先返回再加1,另一个是先加1再返回。