Java Unsafe类的使用

Unsafe类的作用

Unsafe类是rt.jar包中的类,它提供了原子级别的操作,它的方法都是native方法,通过JNI访问本地的C++库。它的出现是为了解决在高并发下的数据同步问题。synchronize关键字修饰的代码块被加了独占锁,相同时间只能有一个线程访问其中的内容,在高并发下这必然会导致大量线程挂起,从用户态转为内核态,带来很大的消耗而且操作效率低下。CAS算法的出现使得在不使用synchronize这种“悲观锁”依然可以实现数据的安全访问,CAS算法是指先读取要修改的变量值,对它进行计算,然后执行检查并更新这个步骤(更新前判断那个值是否是之前那个读到的值),检查并更新是个原子操作,它由硬件来保证可靠性,CAS算法的最后一步的方法就是Unsafe类的方法。因此,Unsafe类的方法可以说是Java高并发的各种扩展类的基础,他们的底层都是调用Unsafe类的方法,Unsafe类为各种扩展类提供底层的原子操作。

Unsafe类的加载和使用

Unsafe类是rt.jar包中的类,rt包是通过顶层的类加载器Bootstrap类加载器加载的。如果不是在Bootstrap类加载器加载的类中试图加载这个类,JVM就会报错。比如我尝试在自己创建的TestUnsafe类中用Unsafe使用getUnsafe()方法获取实例,就会报错。如果一定要使用它,需要使用万能的反射机制。代码如下:

public class TestUnsafe {

//获取Unsafe实例

static final Unsafe unsafe;

//记录state在类TestUsafe中的偏移值

static final long stateOffset;

//变量

public volatile long result=0;

public int[] arr={1,2,3,4,5,6};

static{

try{

//获取成员变量

Field field=Unsafe.class.getDeclaredField("theUnsafe");

//设置为可访问

field.setAccessible(true);

//是静态字段,用null来获取Unsafe实例

unsafe=(Unsafe)field.get(null);

//获取state变量在类中的偏移值

stateOffset=unsafe.objectFieldOffset(TestUnsafe.class.getDeclaredField("result"));

}catch(Exception e){

System.out.println(e.getLocalizedMessage());

throw new Error(e);

}

}

public static void main(String[] args){

TestUnsafe testUnsafe=new TestUnsafe();

//执行并返回结果

for(int i=0;i<1000;i++)

unsafe.getAndAddLong(testUnsafe,stateOffset ,3L);

System.out.println(testUnsafe.result);

System.out.println(unsafe.arrayBaseOffset(testUnsafe.arr.getClass()));

System.out.println(unsafe.arrayIndexScale(testUnsafe.arr.getClass()));

System.out.println(unsafe.compareAndSwapLong(testUnsafe, stateOffset, 3000, 4000));

System.out.println(unsafe.getLongVolatile(testUnsafe, stateOffset));

unsafe.putLongVolatile(testUnsafe, stateOffset, 5000);

System.out.println(testUnsafe.result);

unsafe.putOrderedLong(testUnsafe, stateOffset, 5500);

System.out.println(testUnsafe.result);

Thread thread1=new Thread(){

public void run(){

System.out.println("线程1开始沉睡");

long start=System.currentTimeMillis();

long end=System.currentTimeMillis()+8000;

unsafe.park(true,end);

System.out.println("主线程在"+(System.currentTimeMillis()-start)+"ms后被线程2唤醒");

}

};

Thread thread2=new Thread(){

public void run(){

try {

sleep(3000);

unsafe.unpark(thread1);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

};

thread1.start();

thread2.start();

}

}

输出 :

分析

:在上面的代码中我做了这样一件事。在题目中用static申明了Unsafe实例,long类型的变量偏移量,可见的long变量result。在一段static代码中使用反射获取Unsafe的实例,设置为可访问并使用实例来获取result变量的偏移量,因为这些是静态的,所以在类加载的时候这些事情都做完了。接下来我调用了unsafe的

getAndAddLong(testUnsafe,stateOffset

,3L);方法对变量reslut进行了1000次的加3操作,最终输出结果是3000。

unsafe.getAndAddLong(testUnsafe,stateOffset ,3L);方法

:这是一个原子操作,输入是参数是所在的类对象,变量在内存中的偏移量,加多少(3L表示加3,L表示这是一个long类型的变量)。“偏移量”这个参数就足够体现出这是一个原子操作了,直接对指定位置赋值。

stateOffset=unsafe.objectFieldOffset(TestUnsafe.class.getDeclaredField(“result”));方法

:这个方法是获取指定成员变量在内存中的偏移地址,参数是Field,这里必须通过反射机制获取。

unsafe.arrayBaseOffset(testUnsafe.arr.getClass())方法 :获取数组中第一个元素的地址。

unsafe.arrayIndexScale(testUnsafe.arr.getClass())方法 :获取数组中一个元素占用的字节。

unsafe.compareAndSwapLong(testUnsafe, stateOffset, 3000, 4000)方法

:比较并更新。比较指定偏移量下的变量值是否和3000这个expect值相等,如果相等则把它更新为4000,这是原子操作,由硬件提供可靠性。

unsafe.getLongVolatile(testUnsafe, stateOffset)方法

:获取指定对象中某偏移量下的volatile值,这里需要输入一个对象是因为,在多个对象的情况下,每个对象都是被分配存储空间的,因此,指定对象下的某个变量才是唯一的。

unsafe.putLongVolatile(testUnsafe, stateOffset, 5000);方法

:设置指定对象某偏移量下的long类型的field值为5000,支持volatile语义。

unsafe.putOrderedLong(testUnsafe, stateOffset, 5500);方法

:作用和上面这个方法一样,只是它是一个有延迟的putLongVolatile方法,对变量的修改不会对别的线程立刻可见,一般在希望在这期间变量被意外修改才使用它。

unsafe.park(true,end);方法

:阻塞当前线程一段时间。true表示绝对时间(单位ms),false表示相对时间(单位ns),当false和0作为输入时表示一直阻塞。

unsafe.unpark(thread1);方法 :唤醒指定线程。thread1.interrupt();也可以达到相同的效果。

原文链接:https://blog.csdn.net/mayifan_blog/article/details/86777656

java unsafe 类_Java Unsafe类的使用相关推荐

  1. java filereader类_Java FileReader类

    FileReader类从InputStreamReader类继承而来.该类按字符读取流中数据.可以通过以下几种构造方法创建需要的对象. 在给定从中读取数据的 File 的情况下创建一个新 FileRe ...

  2. java复用类_java复用类

    1. toString() 每一个非基本类型都有一个toString()方法:当编译器需要从对象获取一个string时,该对象的toString()方法就会被调用. 示例: class WaterSo ...

  3. java高级类_Java高级类特性(一)

    权限类内同包不同包子类不同包非子类 private √ × × × default √ √ × × protected √ √ √ × public √ √ √ √ 四.super关键字的使用 pac ...

  4. java 根据类名示例化类_Java即时类| from()方法与示例

    java 根据类名示例化类 即时类from()方法 (Instant Class from() method) from() method is available in java.time pack ...

  5. java 根据类名示例化类_Java即时类| EpochSecond()方法的示例

    java 根据类名示例化类 EpochSecond()方法的即时类 (Instant Class ofEpochSecond() method) Syntax: 句法: public static I ...

  6. java 大数类_Java大数类介绍

    java能处理大数的类有两个高精度大整数BigInteger和高精度浮点数BigDecimal,这两个类位于java.math包内,要使用它们必须在类前面引用该包:import java.math.B ...

  7. java 根据类名示例化类_Java即时类| plusMillis()方法与示例

    java 根据类名示例化类 即时类plusMillis()方法 (Instant Class plusMillis() method) plusMillis() method is available ...

  8. java 根据类名示例化类_Java LocalDateTime类| atOffset()方法与示例

    java 根据类名示例化类 LocalDateTime类atOffset()方法 (LocalDateTime Class atOffset() method) atOffset() method i ...

  9. java 根据类名示例化类_Java MathContext类| 带示例的getRoundingMode()方法

    java 根据类名示例化类 MathContext类的getRoundingMode()方法 (MathContext Class getRoundingMode() method) getRound ...

最新文章

  1. 双流棠湖中学怎么样_全川前十高中,棠湖中学后来居上,七中、四中、成外是“大佬”...
  2. 用Python开始机器学习(3:数据拟合与广义线性回归)
  3. 干货:不同场景容器内获取客户端源IP的方法
  4. 中级工程师考试2019——地图制图与地理信息系统
  5. 小甲鱼python二_小甲鱼python第二讲课后习题
  6. vs2019轻松完成python项目创建
  7. ns3学习之初识ns3
  8. 模糊C均值聚类算法---图像分割(python)
  9. 惠普打印机USB安装成功,但断开USB重新连接时显示脱机无法打印问题(上一版本的驱动程序还在内存中,因此无法加载驱动程序)
  10. 【国外网站软件下载慢解决】
  11. 混合现实:手柄定位不准或者经常性丢失
  12. 维基解密再爆猛料:CIA利用漏洞入侵全球数十亿个人电子设备
  13. 训练CV模型新思路来了:用NLP大火的Prompt替代微调,性能全面提升
  14. 大学计算机基础与实训教程第,大学计算机基础与实训教程
  15. 表达式的LenB(123程序设计ABC)的值是27吗
  16. iframe不起作用原因
  17. VHDL实现智能药盒
  18. 习题 6.3 输入10个整数,将其中最小的数与第一个数对换,把最大的数与最后一个数对换。写3个函数:1.输入10个数;2.进行处理;3.输出10个数。
  19. 在Linux服务器上安装MySQL并配置,远程连接
  20. 思科 Cisco Packet Tracer 标准ACL的配置

热门文章

  1. 怎样高枕无忧的使用阻力线
  2. 测试管理杂谈之理论篇(转)
  3. 自己动手做QQ木马-----HOOK篇
  4. linux磁盘及文件系统之四swap文件系统
  5. 摩根大通CEO吉米·戴蒙质疑比特币2100万枚上限的道理和错误
  6. 产品更新频繁线下培训成本高?华为云会议助力企业降本增效
  7. struct model in Python
  8. 2017年你不得不知的学习诀窍
  9. 记一次,1个月前的瓜子二手车面试经历
  10. 阿里云 OSS 云存储 文件上传