java UnSafe类探索

UnSafe在jdk8以及之前是一个未公开类,jdk9后开始公开使用,对此我们不做过多探讨。
java并发工具中底层广泛使用了UnSafe,经常使用的方法是compareAndSwapInt,compareAndSwapLong,这两个方法都使用了CAS算法

import sun.misc.Unsafe;

public class TT {
	private static Unsafe unsafe;
	public int test;
	static {
		Field f;
		try {
			f = Unsafe.class.getDeclaredField("theUnsafe");
			f.setAccessible(true);
			unsafe = (Unsafe) f.get(null);
		} catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) {
			e.printStackTrace();
		}
	}
	
	public static void main(String[] args) throws NoSuchFieldException, SecurityException {
		TT t = new TT();
		t.test = 1;
		//求出test这个变量内存中在该对象中的偏移量
		long offset = unsafe.objectFieldOffset(TT.class.getDeclaredField("test"));
		System.out.println(TT.compareAndSwapInt(t, offset, 1, 2));
		System.out.println(t.test);
	}
	
	/** 
	*	比较并更新对象的某一个对象类型的域 
	*@param obj 被操作的对象 
	*@param fieldoffset 被操作的域在对象中的偏移量 
	*@param expect 域的期望值 
	*@param update 域的更新值 
	*/  
	static boolean compareAndSwapInt(Object obj,long fieldoffset, int expect, int update) {
		return unsafe.compareAndSwapInt(obj, fieldoffset, expect, update);
	}
}

unsafe.getAndAddLong(testUnsafe,stateOffset ,3L); 方法:这是一个原子操作,输入是参数是所在的类对象,变量在内存中的偏移量,加多少(3L表示加3,L表示这是一个long类型的变量)。“偏移量”这个参数就足够体现出这是一个原子操作了,直接对指定位置赋值。
stateOffset=unsafe.objectFieldOffset(TestUnsafe.class.getDeclaredField(“result”)); 方法:这个方法是获取指定成员变量在内存中的偏移地址,参数是Field,这里必须通过反射机制获取。
unsafe.arrayBaseOffset(testUnsafe.arr.getClass()) 方法:获取数组中第一个元素的地址。
unsafe.arrayIndexScale(testUnsafe.arr.getClass()) 方法:获取数组中一个元素占用的字节。
unsafe.compareAndSwapLong(testUnsafe, stateOffset, 3000, 4000) 方法:比较并更新。比较指定偏移量下的变量值是否和3000这个expect值相等,如果相等则把它更新为4000,这是原子操作,由硬件提供可靠性。
unsafe.getLongVolatile(testUnsafe, stateOffset) 方法:获取指定对象中某偏移量下的volatile值,这里需要输入一个对象是因为,在多个对象的情况下,每个对象都是被分配存储空间的,因此,指定对象下的某个变量才是唯一的。
unsafe.putLongVolatile(testUnsafe, stateOffset, 5000); 方法:设置指定对象某偏移量下的long类型的field值为5000,支持volatile语义。
unsafe.putOrderedLong(testUnsafe, stateOffset, 5500); 方法:作用和上面这个方法一样,只是它是一个有延迟的putLongVolatile 方法,对变量的修改不会对别的线程立刻可见,一般在希望在这期间变量被意外修改才使用它。
unsafe.park(true,end); 方法:阻塞当前线程一段时间。true表示绝对时间(单位ms),false表示相对时间(单位ns),当false和0作为输入时表示一直阻塞。
unsafe.unpark(thread1); 方法:唤醒指定线程。thread1.interrupt();也可以达到相同的效果。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值