`
qianshangding
  • 浏览: 124694 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

并行编程(1) - sum.msic.Unsafe 一

阅读更多

相信看过java源码的同学,对 sum.msic.Unsafe 这个类并不陌生,特别是在java.util.concurrent包有很多的使用。


sum.msic.Unsafe源码: http://www.docjar.com/html/api/sun/misc/Unsafe.java.html

javadoc: http://www.docjar.com/docs/api/sun/misc/Unsafe.html


sum.msic.Unsafe是一个执行低级别(硬件级别的原子操作),不安全操作的方法结合,因为java无法访问到系统底层,所以我们可以看到sum.msic.Unsafe类大部分都是native的本地方法,虽然该类的方法都是pulblic的,但是我们无法在编程中调用Unsafe(在JDK中可以任意调用),该类只能在授信的代码中使用它的实例。


先介绍下CAS(Compare And Swap),按字面的理解就是比较并交换。CAS包含三个操作值:内存地址,新值,预期原值。如果内存地址中的值和预期原值一样,那么则会替换成新值,否则,将不做任何操作。在sum.msic.Unsafe 类中有大量的compareAndSwap***方法,例如:compareAndSwapInt,compareAndSwapLong,compareAndSwapObject等等。

AtomicBoolean类,Unsafe对象初始化:
// setup to use Unsafe.compareAndSwapInt for updates
private static final Unsafe unsafe = Unsafe.getUnsafe();

获取Unsafe实例静态方法:

public static Unsafe getUnsafe() {
      Class cc = sun.reflect.Reflection.getCallerClass(2);
      if (cc.getClassLoader() != null)
            throw new SecurityException("Unsafe");
      return theUnsafe;
 }

接下来简单介绍一下sum.msic.Unsafe的一些方法:
//扩充内存
public native long reallocateMemory(long address, long bytes);

//分配内存
public native long allocateMemory(long bytes);

//释放内存
public native void freeMemory(long address);

//在给定的内存块中设置值
public native void setMemory(Object o, long offset, long bytes, byte value);

//从一个内存块拷贝到另一个内存块
public native void copyMemory(Object srcBase, long srcOffset, Object destBase, long destOffset, long bytes);

//获取值,不管java的访问限制,其他有类似的getDouble,getLong,getChar等等
public native Object getObject(Object o, long offset);

//设置值,不管java的访问限制,其他有类似的putDouble,putLong,putChar等等
public native void putObject(Object o, long offset);

//从一个给定的内存地址获取本地指针,如果不是allocateMemory方法的,结果将不确定
public native long getAddress(long address);

//存储一个本地指针到一个给定的内存地址,如果地址不是allocateMemory方法的,结果将不确定
public native void putAddress(long address, long x);

//该方法返回给定field的内存地址偏移量,这个值对于给定的filed是唯一的且是固定不变的
public native long staticFieldOffset(Field f);

//报告一个给定的字段的位置,不管这个字段是private,public还是保护类型,和staticFieldBase结合使用
public native long objectFieldOffset(Field f);

//获取一个给定字段的位置
public native Object staticFieldBase(Field f);

//确保给定class被初始化,这往往需要结合基类的静态域(field)
public native void ensureClassInitialized(Class c);

//可以获取数组第一个元素的偏移地址
public native int arrayBaseOffset(Class arrayClass);

//可以获取数组的转换因子,也就是数组中元素的增量地址。将arrayBaseOffset与arrayIndexScale配合使用, 可以定位数组中每个元素在内存中的位置
public native int arrayIndexScale(Class arrayClass);

//获取本机内存的页数,这个值永远都是2的幂次方
public native int pageSize();

//告诉虚拟机定义了一个没有安全检查的类,默认情况下这个类加载器和保护域来着调用者类
public native Class defineClass(String name, byte[] b, int off, int len, ClassLoader loader, ProtectionDomain protectionDomain);

//定义一个类,但是不让它知道类加载器和系统字典
public native Class defineAnonymousClass(Class hostClass, byte[] data, Object[] cpPatches);

//锁定对象,必须是没有被锁的
public native void monitorEnter(Object o);

//解锁对象
public native void monitorExit(Object o);

//试图锁定对象,返回true或false是否锁定成功,如果锁定,必须用monitorExit解锁
public native boolean tryMonitorEnter(Object o);

//引发异常,没有通知
public native void throwException(Throwable ee);

//CAS,类似的有compareAndSwapInt,compareAndSwapLong,compareAndSwapBoolean,compareAndSwapChar等等。
public final native boolean compareAndSwapObject(Object o, long offset,  Object expected, Object x);

// 该方法获取对象中offset偏移地址对应的整型field的值,支持volatile load语义。类似的方法有getIntVolatile,getBooleanVolatile等等
public native Object getObjectVolatile(Object o, long offset); 

//线程调用该方法,线程将一直阻塞直到超时,或者是中断条件出现。
public native void park(boolean isAbsolute, long time);

//终止挂起的线程,恢复正常.java.util.concurrent包中挂起操作都是在LockSupport类实现的,也正是使用这两个方法,后续将会把java.util.concurrent包相关的源码分析整理出来。
public native void unpark(Object thread);

//获取系统在不同时间系统的负载情况
public native int getLoadAverage(double[] loadavg, int nelems);

//创建一个类的实例,不需要调用它的构造函数、初使化代码、各种JVM安全检查以及其它的一些底层的东西。即使构造函数是私有,我们也可以通过这个方法创建它的实例,对于单例模式,简直是噩梦,哈哈
public native Object allocateInstance(Class cls) throws InstantiationException;

sum.msic.Unsafe类有一个私有的实例:theUnsafe

private static final Unsafe theUnsafe = new Unsafe();
我们可以通过反射机制获取Unsafe实例,下面举一个简单的例子:
public class UnsafeTest {
	private static Unsafe unsafe;

	public static void main(String[] args) throws Exception {
		try {
			//通过反射获取rt.jar下的Unsafe类
			Field field = Unsafe.class.getDeclaredField("theUnsafe");
			field.setAccessible(true);
			unsafe = (Unsafe) field.get(null);
			Integer target = 12;
			//compareAndSwapInt方法的属性分别是:目标对象实例,目标对象属性偏移量,当前预期值,要设的值.
			//compareAndSwapInt方法是通过反射修改对象的值,具体修改对象下面那个值,可以通过偏移量,对象字段的偏移量可以通过objectFieldOffset获取
			System.out.println(unsafe.compareAndSwapInt(target, 12, 11, 10));
			System.out.println(target);
		} catch (Exception e) {
			System.out.println("Get Unsafe instance occur error" + e);
		}
	}
}

运行以上代码,设置Eclipse:
将Windows->Preferences->Java-Complicer->Errors/Warnings->Deprecated and restricted API,中的Forbidden references(access rules)设置为Warning,即可以编译通过。


最后:Unsafe类是没有安全保证,稍有疏忽或是使用不当,都可能造成JVM崩溃。


后续将会整理一下java.util.concurrent包相关类的源码分析,对于Unsafe各个方法的详细应用,将会具体分析。

分享到:
评论

相关推荐

    一篇看懂Java中的Unsafe类

    在阅读AtomicInteger的源码时,看到了这个类:sum.msic.Unsafe,之前从没见过。所以花了点时间研究了下,下面这篇文章主要给大家介绍了关于Java中Unsafe类的相关资料,需要的朋友可以参考借鉴,下面来一起学习学习吧

    MSIC10.9.2.rar

    MSIC10.9.2.rar

    MSIC.zip_MSIC_MSIC FOR DELPHI20_MSIC delphi_delphi XML_mitec_usb

    非常全面的计算机资源管理分析器,以xml的数据格式产生报告

    MSIC.zip_MSI_Overview_NTDDK.d_delphix_d7_msi_Machine.dcu_update

    MSIC, source code for delphi

    msic同步接收

    c#接收MSIC同步包

    C语言头文件 MSI C语言头文件 MSI

    C语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言...

    stm32f10xx.h

    Cortex-M4 Device Peripheral Access Layer Header File. This file contains all the peripheral register's definitions, bits definitions and memory mapping for STM32F4xx devices

    andrpod TAB

    tbPlayMusic = (ToggleButton) findViewById(R.id.tb_msic); mediaPlayer = MediaPlayer.create(this, R.raw.nishiwodeyan);//指定了播放源 tbPlayMusic.setOnCheckedChangeListener(new CompoundButton....

    STM32库文件相关

    1. stm32f10x.h 实现外设的寄存器映射,和所有的中断通道IRQ,新库是stm32f105xc.h 2. core_cm3.h 实现内核的寄存器映射,NVIC,SYSTICK等 3. stm32f1xx.h 存放外设初始化结构体,及初始化函数声明 4. stm32f1xx.c...

    Diamond帝盟S100声卡驱动for windows xp

    Diamond帝盟S100声卡S100声卡驱动for windows xp它提供了高达17条的Direct Sound 8.1的硬件音频流,17条的硬件Wave流(谁说Allegro-1不支持硬件多音频流),17条的Direct Sound 3D 8.1的硬件加速,在高速的DSP实时的...

    一种低功耗测试图形的生成方法 (2013年)

    通过设计一种可配置SIC计数器和种子生成电路,证明了该方法中任意的2个 MSIC图形在任何情况下都是相异的。以国际基准测试电路ISCAS’89为对象,在nangate 45 nm工艺上的仿真实验表明,基于该方法的测试生成电路的...

    Delphi Aes加密控件 MSICS_Trial硬件信息控件

    delphi Aes 加密控件 MSICS_Trial 硬件信息控件 获取本地硬件序列号的2个插件。

    3种模式的按键驱动(cdev misc platform).rar

    mini241o 按键的3种驱动 字符设备 混杂设备 platform驱动

    BASE64Encode编码,jar包(源代码打包)

    BASE64Encode 编码,在andorid没有这个编码,这里面有打成jar包,可以导进去。用源代码打包的。手懒不愿意自己弄得话,可以下载导进去。里面附带着源代码,也可将源代码开进去直接使用,希望能帮助大家。

    通过喷砂技术获得的高性能三相边界,用于混合电位型氧化锆基NO2传感器

    与其他传感器相比,使用80μmSiC处理的YSZ基板的传感器在850°C时对NO2的信号最大,其对100 ppm NO2的响应比基于传感器的传感器高约1.08、1.21和1.5倍在分别用60μmSiC,40μmSiC和光滑的YSZ衬底喷砂处理过的YSZ...

    KeePass Password Safe 2

    ctf中不太常用的一个工具 但有的题会用到 KeePass Password Safe是一款优秀的密码管理工具,实际上可能还不仅仅是密码管理工具,重要的文件也可以保存在这款软件中

    甲骨论-Undo表空间管理

    甲骨论-08-Undo表空间管理 本章详细介绍了undo表空间的工作原理以及与undo相关的数据字典

Global site tag (gtag.js) - Google Analytics