Unsafe类

一、Unsafe类仅能被BootstrapClassLoader加载的类实例化,用户建的类默认都是ApplicationClassLoader加载的,实例化Unsafe时会报错(可直接管理内存,不对用户开放使用)。可以用反射实例化(方式一)。补充:加启动参数指定当前类由BootstrapClassLoader加载(方式二)

    private Unsafe() {//构造方法私有化,与下面static构造单例
    }

    @CallerSensitive
    public static Unsafe getUnsafe() {
        Class var0 = Reflection.getCallerClass();
        if (!VM.isSystemDomainLoader(var0.getClassLoader())) {//判断是否BootstrapClassLoader加载的类执行实例化的
            throw new SecurityException("Unsafe");
        } else {
            return theUnsafe;
        }
    }

    static {
        registerNatives();
        Reflection.registerMethodsToFilter(Unsafe.class, new String[]{"getUnsafe"});
        theUnsafe = new Unsafe();
        ARRAY_BOOLEAN_BASE_OFFSET = theUnsafe.arrayBaseOffset(boolean[].class);
        ARRAY_BYTE_BASE_OFFSET = theUnsafe.arrayBaseOffset(byte[].class);
        ARRAY_SHORT_BASE_OFFSET = theUnsafe.arrayBaseOffset(short[].class);
        ARRAY_CHAR_BASE_OFFSET = theUnsafe.arrayBaseOffset(char[].class);
        ARRAY_INT_BASE_OFFSET = theUnsafe.arrayBaseOffset(int[].class);
        ARRAY_LONG_BASE_OFFSET = theUnsafe.arrayBaseOffset(long[].class);
        ARRAY_FLOAT_BASE_OFFSET = theUnsafe.arrayBaseOffset(float[].class);
        ARRAY_DOUBLE_BASE_OFFSET = theUnsafe.arrayBaseOffset(double[].class);
        ARRAY_OBJECT_BASE_OFFSET = theUnsafe.arrayBaseOffset(Object[].class);
        ARRAY_BOOLEAN_INDEX_SCALE = theUnsafe.arrayIndexScale(boolean[].class);
        ARRAY_BYTE_INDEX_SCALE = theUnsafe.arrayIndexScale(byte[].class);
        ARRAY_SHORT_INDEX_SCALE = theUnsafe.arrayIndexScale(short[].class);
        ARRAY_CHAR_INDEX_SCALE = theUnsafe.arrayIndexScale(char[].class);
        ARRAY_INT_INDEX_SCALE = theUnsafe.arrayIndexScale(int[].class);
        ARRAY_LONG_INDEX_SCALE = theUnsafe.arrayIndexScale(long[].class);
        ARRAY_FLOAT_INDEX_SCALE = theUnsafe.arrayIndexScale(float[].class);
        ARRAY_DOUBLE_INDEX_SCALE = theUnsafe.arrayIndexScale(double[].class);
        ARRAY_OBJECT_INDEX_SCALE = theUnsafe.arrayIndexScale(Object[].class);
        ADDRESS_SIZE = theUnsafe.addressSize();
    }

二、反射实现实例化

public class UnsafeTest2 {
    static final Unsafe unsafe;

    static final long stateOffset;

    private volatile long state = 0;

    static {
        try {
            Field field = Unsafe.class.getDeclaredField("theUnsafe");
            field.setAccessible(true);
            unsafe = (Unsafe)field.get(null);//反射得到Unsafe实例
            stateOffset = unsafe.objectFieldOffset(UnsafeTest.class.getDeclaredField("state"));
        }catch (Exception e){
            System.out.println(e.getLocalizedMessage());
            throw new Error(e);
        }
    }

    public static void main(String[] args){
        UnsafeTest2 test = new UnsafeTest2();
        Boolean sucess = unsafe.compareAndSwapInt(test,stateOffset,0,1);
        System.out.println(sucess);
    }
}

三、重要方法

long objectFieldOffset(Field field):返回指定的变量在所属类中的内存偏移地址

int arrayBaseOffset(Class arrayClass):获取数组中第一个元素的地址。

int arrayIndexScale(Class arrayClass):获取数组中第一个元素占用的字节。

boolean compareAndSwapLong(Object obj,long offset,long expect,long update):比较对象obj中偏移量为offset的变量的值是否与expect相等,相等则使用update更新,然后返回true,否则返回false。

public native long getLongvolatile(Object obj,long offset):获取obj对象中偏移量为offset的变量对应的volatile语义的值。

void putLongvolatile(Object obj,long offset,long value):设置obj对象中offset偏移的类型为long的field的值的value,支持volatile语义(可见性,有序性)

void putOrderedLong(Object obj,long offset,long value):设置obj对象中offset偏移地址对应的long型field的值为value,无volatile语义(可见性,有序性)

void park(boolean isAbsolute,long time):阻塞当前线程,其中参数isAbsolute等于false且time等于0时表示一直阻塞

void unpark(Object thread):唤醒调用park后阻塞的线程。

long getAndSetLong(Object obj,long offset,long update):获取对象obj中偏移量为offset的变量volatile语义的当前值,并设置变量volatile语义的值为update。

long getAndAddLong(Object obj,long offset,long addValue):获取对象obj中偏移量为offset的变量volatile语义的当前值,并设置变量的值为原始值+addValue。

参考自《Java并发编程之美》

posted on 2020-01-06 03:06  FFStayF  阅读(314)  评论(0编辑  收藏  举报