优秀的编程知识分享平台

网站首页 > 技术文章 正文

java unsafe用法解析,int、long、string

nanyue 2024-08-01 22:55:40 技术文章 5 ℃

在阅读java并发工具包源码时,通过CAS通常会遇到unsafe类,那么自己该如何理解unsage类的几个参数含义,以及使用呢?

这里列举一个实例:

实例中列举了int、long、string三种字段的用法。

其中可以看出int占用了4字节,而long占用了8字节,这与java基本类型占用字节数是一致的

1字节: byte , boolean
2字节: short , char
4字节: int , float
8字节: long , double

import sun.misc.Unsafe;

import java.lang.reflect.Field;

class Car {

    int intParam = 3;

    long longParam =5L;

    String strParam;
}
public class UnsafeTest {

    private static Unsafe unsafe;

    static{
        try {
            Field field = Unsafe.class.getDeclaredField("theUnsafe");
            field.setAccessible(true);
            unsafe = (Unsafe)field.get(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) throws Exception {

        Class clazz = Car.class;
        //获取所有声明字段
        Field[] fields = clazz.getDeclaredFields();
        System.out.println("开始测试偏移量。");
        //一个实例存在于内存,不同的属性占用不同的内存空间,偏移量就是这个属性所在空间的内存地址
        for(Field f : fields){
            //获取属性偏移量,可以通过这个偏移量给属性设值
            System.out.println(f.getName()+" : "+unsafe.objectFieldOffset(f));
        }
        System.out.println("--------------------------------");
        Car car = new Car();
        Field intParam = clazz.getDeclaredField("intParam");
        int a = (Integer) intParam.get(car);
        System.out.println("intParam原始值:"+a);

        long intParamOffset = unsafe.objectFieldOffset(intParam);
        System.out.println("intParam实例变量修改前偏移量:"+intParamOffset);
        //intParam实例变量偏移量是offset 原始值是3,我们要改成10
        System.out.println(unsafe.compareAndSwapInt(car, intParamOffset, 3, 10));
        System.out.println("intParam改变之后的值:"+car.intParam);

        long intParamOffsetAfter = unsafe.objectFieldOffset(intParam);
        System.out.println("intParam改变之后偏移量是:"+intParamOffsetAfter);
        System.out.println("------------改为10后,再次调用,返回false--------------------");
        System.out.println(unsafe.compareAndSwapInt(car, intParamOffset, 3, 10));

        System.out.println("------------开始修改long--------------------");
        Field longParam = clazz.getDeclaredField("longParam");
        long longInit = (Long) longParam.get(car);
        System.out.println("longParam原始值:"+longInit);
        long longParamOffset = unsafe.objectFieldOffset(longParam);
        System.out.println("longParam实例变量的偏移量是:"+longParamOffset);
        System.out.println(unsafe.compareAndSwapLong(car, longParamOffset, 5L, 111L));
        System.out.println("longParam改变之后的值:"+car.longParam);

        System.out.println("------------开始修改string--------------------");
        Field strParam = clazz.getDeclaredField("strParam");
        String str = (String) strParam.get(car);
        System.out.println("strParam原始值:"+str);
        long strParamOffset = unsafe.objectFieldOffset(strParam);
        System.out.println("strParam实例变量的偏移量是:"+strParamOffset);
        System.out.println(unsafe.compareAndSwapObject(car, strParamOffset, null, "5"));
        System.out.println("strParam改变之后的值:"+car.strParam);

    }

}

运行结果


最近发表
标签列表