作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO
联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬
学习必须往深处挖,挖的越深,基础越扎实!
阶段1、深入多线程
阶段2、深入多线程设计模式
阶段3、深入juc源码解析
码哥源码部分
码哥讲源码-原理源码篇【2024年最新大厂关于线程池使用的场景题】
码哥讲源码-原理源码篇【揭秘join方法的唤醒本质上决定于jvm的底层析构函数】
码哥源码-原理源码篇【Doug Lea为什么要将成员变量赋值给局部变量后再操作?】
码哥讲源码【谁再说Spring不支持多线程事务,你给我抽他!】
打脸系列【020-3小时讲解MESI协议和volatile之间的关系,那些将x86下的验证结果当作最终结果的水货们请闭嘴】
一、Unsafe简介
在正式的开讲 juc-atomic框架系列之前,有必要先来了解下Java中的Unsafe类。
Unsafe类,来源于sun.misc
包。该类封装了许多类似指针操作,可以直接进行内存管理、操纵对象、阻塞/唤醒线程等操作。Java本身不直接支持指针的操作,所以这也是该类命名为Unsafe的原因之一。
J.U.C中的许多CAS方法,内部其实都是Unsafe类在操作。
比如AtomicBoolean
的compareAndSet
方法:
unsafe.compareAndSwapInt
方法是个native方法。(如果对象中的字段值与期望值相等,则将字段值修改为x,然后返回true;否则返回false):
入参的含义如下:
参数名称 | 含义 |
---|---|
o | 需要修改的对象 |
offset | 需要修改的字段到对象头的偏移量(通过偏移量,可以快速定位修改的是哪个字段) |
expected | 期望值 |
x | 要设置的值 |
Unsafe类中CAS方法都是native方法,需要通过CAS原子指令完成。在讲AQS时,里面有许多涉及CLH队列的操作,其实就是通过Unsafe类完成的指针操作。
二、Unsafe对象的创建
Unsafe是一个final类,不能被继承,也没有公共的构造器,只能通过工厂方法 getUnsafe 获得Unsafe的单例。
但是 getUnsafe 方法限制了调用该方法的类的类加载器必须为 Bootstrap ClassLoader 。
Java中的类加载器可以大致划分为以下三类:
类加载器名称 | 作用 |
---|---|
Bootstrap类加载器(BootstrapClassLoader) | 主要加载的是JVM自身需要的类,这个类加载使用C++语言实现的,是JVM自身的一部分,它负责将【JDK的安装目录】/lib路径下的核心类库,如rt.jar |
扩展类加载器(ExtensionClassLoader) | 该加载器负责加载【JDK的安装目录】\jre\lib\ext目录中的类库,开发者可以直接使用该加载器 |
系统类加载器(ApplicationClassLoader) | 负责加载用户类路径(ClassPath)所指定的类,开发者可以直接使用该类加载器,也是默认的类加载器 |
所以在用户代码中直接调用 getUnsafe 方法,会抛出异常。因为用户自定义的类一般都是由系统类加载器加载的。
但是,是否就真的没有办法获取到Unsafe实例了呢?当然不是,要获取Unsafe对象的方法很多,这里给出一种通过反射的方法:
Field f = Unsafe.class.getDeclaredField("theUnsafe");
f.setAccessible(true);
Unsafe unsafe = (Unsafe) f.get(null);
但是,除非对Unsafe的实现非常清楚,否则应尽量避免直接使用Unsafe来进行操作。