美文网首页java
jdk中的底层类库(Unsafe)

jdk中的底层类库(Unsafe)

作者: suxin1932 | 来源:发表于2019-08-01 09:26 被阅读0次

1.sun.misc.Unsafe

Unsafe类是在sun.misc包下,不属于Java标准。

但是很多Java的基础类库,包括一些被广泛使用的高性能开发库都是基于Unsafe类开发的
比如Netty、Hadoop、Kafka等。

使用Unsafe可用来直接访问系统内存资源并进行自主管理,
Unsafe类在提升Java运行效率,增强Java语言底层操作能力方面起了很大的作用。

Unsafe可认为是Java中留下的后门,提供了一些低层次操作,如直接内存访问、线程调度等。

1.1Unsafe的大部分API都是native的方法

1)Class相关。
主要提供Class和它的静态字段的操作方法。
staticFieldOffset(静态域偏移)、defineClass(定义类)、defineAnonymousClass(定义匿名类)、
ensureClassInitialized(确保类初始化)、objectFieldOffset(对象域偏移)等方法。
通过这些方法我们可以获取对象的指针,通过对指针进行偏移,
我们不仅可以直接修改指针指向的数据(即使它们是私有的),
甚至可以找到JVM已经认定为垃圾、可以进行回收的对象。

2)Object相关。
主要提供Object和它的字段的操作方法。
allocateInstance()方法提供了另一种创建实例的途径。
通常我们可以用new或者反射来实例化对象,
使用allocateInstance()方法可以直接生成对象实例,且无需调用构造方法和其它初始化方法。
这在对象反序列化的时候会很有用,能够重建和设置final字段,而不需要调用构造方法。

3)Arrray相关。
主要提供数组及其中元素的操作方法。
包括了arrayBaseOffset(获取数组第一个元素的偏移地址)、
arrayIndexScale(获取数组中元素的增量地址)等方法。
arrayBaseOffset与arrayIndexScale配合起来使用,就可以定位数组中每个元素在内存中的位置。
由于Java的数组最大值为Integer.MAX_VALUE,使用Unsafe类的内存分配方法可以实现超大数组。
实际上这样的数据就可以认为是C数组,因此需要注意在合适的时间释放内存。

4)并发相关。
主要提供低级别同步原语,如CAS、线程调度、volatile、内存屏障等。
>>>>>>>>>>>> CAS
Unsafe类的CAS操作可能是用的最多的,它为Java的锁机制提供了一种新的解决办法,
比如AtomicInteger等类都是通过该方法来实现的。
compareAndSwap方法是原子的,可以避免繁重的锁机制,提高代码效率。
这是一种乐观锁,通常认为在大部分情况下不出现竞态条件,如果操作失败,会不断重试直到成功。
>>>>>>>>>>>> LockSupport: park(挂起线程) & unpark(恢复线程)
将一个线程进行挂起是通过park方法实现的,调用 park后,线程将一直阻塞直到超时或者中断等条件出现。
unpark可以终止一个挂起的线程,使其恢复正常。
整个并发框架中对线程的挂起操作被封装在LockSupport类中,最终都调用了Unsafe.park()方法。
>>>>>>>>>>>> 内存屏障
包括了loadFence、storeFence、fullFence等方法。
这是在Java 8新引入的,用于定义内存屏障,避免代码重排序。
loadFence() 表示该方法之前的所有load操作在内存屏障之前完成。
storeFence()表示该方法之前的所有store操作在内存屏障之前完成。
fullFence()表示该方法之前的所有load、store操作在内存屏障之前完成。

5)Memory相关。
提供了直接内存访问方法(绕过Java堆直接操作本地内存),可做到像C一样自由利用系统内存资源。
用Unsafe开辟的内存空间不占用Heap空间,当然也不具有自动内存回收功能。
该部分包括了
allocateMemory(分配内存, 该分配的内存需要手动free, 不被GC回收)、
reallocateMemory(重新分配内存)、copyMemory(拷贝内存)、
freeMemory(释放内存 )、getAddress(获取内存地址)、addressSize、pageSize、
getInt(获取内存地址指向的整数)、getIntVolatile(获取内存地址指向的整数,并支持volatile语义)、
putInt(将整数写入指定内存地址)、putIntVolatile(将整数写入指定内存地址,并支持volatile语义)、
putOrderedInt(将整数写入指定内存地址、有序或者有延迟的方法)等方法。
getXXX和putXXX包含了各种基本类型的操作。
利用copyMemory方法,我们可以实现一个通用的对象拷贝方法,
无需再对每一个对象都实现clone方法,当然这通用的方法只能做到对象浅拷贝。

6)系统相关。
主要返回某些低级别的内存信息,如地址大小、内存页大小。

参考资源
https://www.jb51.net/article/140726.htm
https://www.jb51.net/article/140709.htm
http://mishadoff.com/blog/java-magic-part-4-sun-dot-misc-dot-unsafe/

相关文章

网友评论

    本文标题:jdk中的底层类库(Unsafe)

    本文链接:https://www.haomeiwen.com/subject/azbfdctx.html