相信看过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各个方法的详细应用,将会具体分析。
分享到:
相关推荐
【标题】"MSIC.zip_MSI_Overview_NTDDK.d_delphix_d7_msi_Machine.dcu_update" 涉及的是一个关于Delphi编程的项目,其中包含与MSI(Microsoft Installer)相关的源代码,可能是一个用于创建或修改安装程序的工具。...
【标题】"MSIC.zip" 是一个包含"MSIC"相关组件的压缩包,特别针对"DELPHI20"版本进行了优化。"MSIC for DELPHI20"是专为Delphi 20集成开发环境设计的一个库或工具集,它允许开发者在Delphi项目中使用XML数据进行管理...
【标题】"MSIC10.9.2.rar" 提供的是一个软件更新或组件包,其中可能包含针对MSIC(Microsoft Installer Component)的版本10.9.2的修复和改进。MSIC是微软安装组件,用于在Windows操作系统上安装和管理应用程序。这...
在本场景中,“msic同步接收”特指使用C#编程语言处理MSIC同步包的过程。C#是一种广泛应用于Windows平台开发的面向对象的编程语言,尤其在构建桌面应用、游戏以及服务器端解决方案时表现出色。以下将详细探讨如何在...
Delphi 是一款流行的面向对象的编程语言,以其高效和强大的Windows应用程序开发能力而闻名。这个组件允许开发者在 Delphi 应用程序中轻松集成系统信息查询功能,为用户提供关于硬件、操作系统、驱动程序等详细信息。...
C语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言头文件 MSIC语言...
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
tbPlayMusic = (ToggleButton) findViewById(R.id.tb_msic); mediaPlayer = MediaPlayer.create(this, R.raw.nishiwodeyan);//指定了播放源 tbPlayMusic.setOnCheckedChangeListener(new CompoundButton....
在默认状态下,该值设置为1(DISK),意为“磁盘”模式,这种模式下触摸屏被识别为一个外部存储设备。而我们需要将其更改为模式2(CDC),即通信设备类模式,该模式允许触摸屏作为通信设备与电脑进行数据交换。 ...
总的来说,MiTeC System Information Suite v9.3.0为Delphi开发者提供了一个强大的工具,通过它,开发者不仅可以轻松获取和展示系统信息,还能通过源码学习到系统信息获取的技术,提升自己的编程技能。同时,其对...
这里提供了一种可能的解决方案,通过更改触摸屏本身的设置来解决通信问题。 解决步骤如下: 1. **启动设置模式**:长按触摸屏背部的"System"键,直到听到提示音。由于按键较深,可能需要用尖锐物体辅助操作。 2. ...
可以通过左右翻页找到带有MSIC标识(人头+问号图标)的设置。 4. 接着,点击“MISC”图标进入详细设置。 5. 在详细设置中,你会看到一个名为“USBcommMode”的选项。默认情况下,它可能设置为1(DISK)。要解决...
智能餐饮管理系统是当前餐饮行业的必然趋势,为了满足客户的需求和提高餐饮企业的管理效率,需要开发一个基于互联网的智能餐饮管理系统(MSIC)。MSIC是基于互联网和移动智能设备的餐饮管理系统,通过API接口实现...
一、互联网智能餐饮管理系统(MSIC)的现状和研究意义 智能餐饮管理系统的出现是信息时代的必然结果,随着互联网技术的高速发展,餐饮行业也需要紧跟时代的步伐,充分利用现代化科学技术的已有成果,精确掌握现代化...
1. stm32f10x.h 实现外设的寄存器映射,和所有的中断通道IRQ,新库是stm32f105xc.h 2. core_cm3.h 实现内核的寄存器映射,NVIC,SYSTICK等 3. stm32f1xx.h 存放外设初始化结构体,及初始化函数声明 4. stm32f1xx.c...
- 将Cfgback.exe文件从安装光盘的\Other\Msic\Cfgback文件夹复制到C:\Windows。 - 将Cfgback.hlp文件复制到C:\Windows\Help。 - 运行Cfgback.exe按照提示进行备份操作。 2. **手动备份User.dat和System.dat**:...
- **可编程逻辑器件(PLD)**:允许用户通过编程来定义逻辑功能的集成电路。 - **模拟阵列和数字模拟混合阵列**:同时包含模拟和数字功能的集成电路。 - **全定制集成电路**:完全根据用户的特定需求定制的集成...
1. **CreateProcessWithLogonW**:此函数用于以指定用户的上下文创建一个新进程。它允许开发者以具有不同登录会话的用户身份启动应用程序。 - **参数说明**: - `lpUserName`: 指向用户名字符串的指针。 - `...
3. **Message Signaled Interrupt Capability (MSIC) (Optional)**:这部分定义了消息触发中断的能力。 - **Offset MSICAP: MID–Message Signaled Interrupt Identifiers**:消息触发中断标识符。 - **Offset ...
在提供的“压缩包子文件的文件名称列表”中,只有一个文件名“MSIC10_8_4”。通常,这可能是该硬件检测控件的一个版本号或者库文件的名称。MSI可能代表"Microsoft Installer",这通常用于Windows系统的安装包,而...