Unsafe方法学习

介绍

unsafe类主要是用于提供一些用于执行低级别、不安全操作的方法,如直接访问系统内存资源、自主管理内存资源等,这些方法在提升Java运行效率,增强Java语言底层资源操作能力方面起到了很大的作用。

unsafe提供的这些功能实现需要依赖本地方法。是通过Java中其他编程语言编写的方法。

Unsafe的创建

java 复制代码
private Unsafe() {
}

@CallerSensitive
public static Unsafe getUnsafe() {
    Class var0 = Reflection.getCallerClass();
    if (!VM.isSystemDomainLoader(var0.getClassLoader())) {
        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();
    }

Unsafe的创建是单例模式。采用的是饿汉式。

Unsafe方法的使用

java 复制代码
@CallerSensitive
public static Unsafe getUnsafe() {
    Class var0 = Reflection.getCallerClass();
    if (!VM.isSystemDomainLoader(var0.getClassLoader())) {
        throw new SecurityException("Unsafe");
    } else {
        return theUnsafe;
    }
}

在代码中直接调用getUnsafe方法,会对调用者的classLoader进行检查,判断当前类是否由Bootstrap classLoader加载,如果不是就抛出SecurityException异常。只有启动类加载器才能够调用Unsafe类中的方法,防止这些代码在不可信的代码中被调用。

获取实例

  1. 利用反射获取Unsafe类中已经实例化完成的单例对象theUnsafe。
java 复制代码
Field field = Unsafe.class.getDeclaredField("theUnsafe");
field.setAccessible(true);
Unsafe unsafe = (Unsafe) field.get(null);
  1. 通过Java命令执行-Xbootclasspath/a把调用Unsafe相关方法的类A所在jar包路径追加到默认的bootstrap路径中,使得A被引导类加载器加载,从而通过Unsafe.getUnsafe发放安全的获取Unsafe实例。
java 复制代码
java -Xbootclasspath/a: ${path}   // 其中path为调用Unsafe相关方法的类所在jar包路径

Unsafe功能

Unsafe功能如下,可以被分为8类:

  1. 内存操作

  2. 内存屏障

  3. 对象操作

  4. 数组操作

  5. CAS操作
    CAS是CPU的原子指令,不会造成所谓的数据不一致问题,Unsafe提供的CAS方法底层实现即为CPU指令。
    在Unsafe类中,提供了compareAndSwapObject、compareAndSwapInt、compareAndSwapLong方法来实现对Object、int、long类型的CAS操作。
    以compareAndSwapInt方法为例:

    java 复制代码
    public final native boolean compareAndSwapInt(Object o, long offset,int expected,int x);

    参数o为需要更新的对象,offset是对象o中整形字段的偏移量,如果这个字段的值与excepted相同,则将字段的值设为x这个新值,并且此更新是不可被中断的,是一个原子操作。
    注: 被修改的对象的字段类型需要是基本数据类型。

  6. 线程调度

  7. Class操作

  8. 系统信息

参考

https://blog.csdn.net/weixin_46058921/article/details/143012338

相关推荐
sensen_kiss4 分钟前
CPT304 SoftwareEngineeringII 软件工程 2 Pt.9 软件测试 (Software Testing)(下)
学习·软件工程
会Tk矩阵群控的小木8 分钟前
基于Python的iMessage短信群发与社媒多账号统一管理系统实现
开发语言·windows·python·新媒体运营·开源软件·个人开发
程序员黑豆10 分钟前
AI全栈开发 - Java:变量
java·前端·ai编程
我是一颗柠檬10 分钟前
【Java项目技术亮点】分库分表+数据路由策略:单表5000万后的架构升级方案
java·开发语言·分布式·架构
wu_ye_m12 分钟前
学习c语言第35天 函数声明和定义
c语言·开发语言·学习
布朗克16819 分钟前
25 IO流高级操作——序列化、NIO与Files工具类
java·数据库·io·nio
小研说技术24 分钟前
Spring AI实现rag流程(简易版)
java·后端
清辞85326 分钟前
Coze从入门到实战---第一、二章
大数据·人工智能·学习·语言模型
亓才孓29 分钟前
【本地项目引用外部库的类,想修改字段遇到的请缓存的问题】
java·maven
质造者34 分钟前
LangChain + Ollama + Tavily 实现旅游问答系统
linux·人工智能·python·langchain·rag