java Unsafe类中compareAndSwap相关介绍

转自
https://blog.csdn.net/sherld/article/details/42492259
https://blog.csdn.net/gol_phing/article/details/48931727

https://blog.csdn.net/sherld/article/details/42492259
这篇文章的内容:
最近在看jdk7中java.util.concurrent下面的源码中,发现许多类中使用了Unsafe类中的方法来保证并发的安全性,而java 7 api中并没有这个类的相关介绍,在网上查了许多资料,其中http://ifeve.com/sun-misc-unsafe/这个网站详细的讲解了Unsafe的相关用法,而下面是结合网站中的介绍和具体的AtomicInteger类来讲解一下其相关的用法。

private static final Unsafe unsafe = Unsafe.getUnsafe();
private static final long valueOffset;
 
static {
    try {
        valueOffset = unsafe.objectFieldOffset
            (AtomicInteger.class.getDeclaredField("value"));
    } catch (Exception ex) { throw new Error(ex); }
}
 
private volatile int value;

首先可以看到AtomicInteger类在域中声明了这两个私有变量unsafe和valueOffset。其中unsafe实例采用Unsafe类中静态方法getUnsafe()得到,但是这个方法如果我们写的时候调用会报错,因为这个方法在调用时会判断类加载器,我们的代码是没有“受信任”的,而在jdk源码中调用是没有任何问题的;valueOffset这个是指类中相应字段在该类的偏移量,在这里具体即是指value这个字段在AtomicInteger类的内存相对于该类首地址的偏移量

然后可以看一个有一个静态初始化块,这个块的作用即是求出value这个字段的偏移量。具体的方法使用的反射的机制得到value的Field对象,再根据objectFieldOffset这个方法求出value这个变量内存中在该对象中的偏移量

volatile关键字保证了在多线程中value的值是可见的,任何一个线程修改了value值,会将其立即写回内存当中

public final int getAndSet(int newValue) {
	for (;;) {
		int current = get();
		if (compareAndSet(current, newValue))
			return current;
	}
}
 
public final boolean compareAndSet(int expect, int update) {
	return unsafe.compareAndSwapInt(this, valueOffset, expect, update);
}

getAndSet这个方法作用为将value值设置为newValue,并返回修改前的value值。
在for循环中,保证每次如果compareAndSet这个方法失败之后,能重新进行尝试,直到成功将value值设置为newValue。

compareAndSet这个方法主要调用unsafe.compareAndSwapInt这个方法,这个方法有四个参数,其中第一个参数为需要改变的对象,第二个为偏移量(即之前求出来的valueOffset的值,确定要修改哪个变量),第三个参数为期待的值,第四个为更新后的值。整个方法的作用即为若调用该方法时,value的值与expect这个值相等,那么则将value修改为update这个值,并返回一个true,如果调用该方法时,value的值与expect这个值不相等,那么不做任何操作,并范围一个false。

因此之所以在getAndSet方法中调用一个for循环,即保证如果调用compareAndSet这个方法返回为false时,能再次尝试进行修改value的值,直到修改成功,并返回修改前value的值

整个代码能保证在多线程时具有线程安全性并且没有使用java中任何锁的机制,所依靠的便是Unsafe这个类中调用的该方法具有原子性,这个原子性的保证并不是靠java本身保证,而是靠一个更底层的与操作系统相关的特性实现


https://blog.csdn.net/gol_phing/article/details/48931727
这篇文章的内容:

compareAndSwap是个原子方法,原理是cas。就是说如果他是xx,那么就改为xxx。
这个是高效,而且是原子的,不用加锁。
但是其他值改了而产生误操作,应为会先判断当前值,符合期望才去改变。

 package com.huangyunbin;  
    import java.lang.reflect.Field;  
    import sun.misc.Unsafe;  
      
      
    public class UnsafeTest {  
      
        private static Unsafe unsafe;  
      
        static {  
            try {  
                //通过反射获取rt.jar下的Unsafe类  
                Field field = Unsafe.class.getDeclaredField("theUnsafe");  
                field.setAccessible(true);  
                unsafe = (Unsafe) field.get(null);  
            } catch (Exception e) {  
                System.out.println("Get Unsafe instance occur error"+ e);  
            }  
        }  
      
      
      
      
        public static void main(String[] args) throws Exception  
        {  
            Class clazz = Target.class;  
            Field[] fields = clazz.getDeclaredFields();  
            System.out.println("fieldName:fieldOffset");  
            for (Field f : fields) {  
                // 获取属性偏移量,可以通过这个偏移量给属性设置  
                System.out.println(f.getName() + ":" + unsafe.objectFieldOffset(f));  
            }  
            Target target = new Target();  
            Field intFiled  =  clazz.getDeclaredField("intParam")  ;  
            int a=(Integer)intFiled.get(target ) ;  
            System.out.println("原始值是:"+a);  
            //intParam的字段偏移是12 原始值是3 我们要改为10  
            System.out.println(unsafe.compareAndSwapInt(target, 12, 3, 10));  
            int b=(Integer)intFiled.get(target) ;  
            System.out.println("改变之后的值是:"+b);  
      
            //这个时候已经改为10了,所以会返回false  
            System.out.println(unsafe.compareAndSwapInt(target, 12, 3, 10));  
      
            System.out.println(unsafe.compareAndSwapObject(target, 24, null, "5"));  
        }  
    }  
      
      
     class Target {  
         int intParam=3;  
         long longParam;  
         String strParam;  
         String strParam2;  
    }  

运行结果:

fieldName:fieldOffset 
intParam:12 
longParam:16 
strParam:24 
strParam2:28 
原始值是:3 
true 
改变之后的值是:10 
false 
true 

compareAndSwapInt是通过反射根据字段偏移去修改对象的。
可以看到int是4个字节的偏移量,long是4个字节的偏移量,string是4个字节的偏移量 。
注意 Unsafe的对象不能直接new,要通过反射去获取

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值