使用sun.misc.Unsafe

1)、突破限制创建实例

    通过allocateInstance()方法,你可以创建一个类的实例,但是却不需要调用它的构造函数、初使化代码、各种JVM安全检查以及其它的一些底层的东西。即使构造函数是私有,我们也可以通过这个方法创建它的实例。

    (这个对单例模式情有独钟的程序员来说将会是一个噩梦,它们没有办法阻止这种方式调用大笑

    看下面一个实例(注:为了配合这个主题,译者将原实例中的public构造函数修改为了私有的):

 

  1. public class UnsafeDemo {  
  2.     public static void main(String[] args) throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException, InstantiationException {  
  3.         Field f = Unsafe.class.getDeclaredField("theUnsafe"); // Internal reference  
  4.         f.setAccessible(true);  
  5.         Unsafe unsafe = (Unsafe) f.get(null);  
  6.   
  7.         // This creates an instance of player class without any initialization  
  8.         Player p = (Player) unsafe.allocateInstance(Player.class);  
  9.         System.out.println(p.getAge()); // Print 0  
  10.   
  11.         p.setAge(45); // Let's now set age 45 to un-initialized object  
  12.         System.out.println(p.getAge()); // Print 45  
  13.     }  
  14. }  
  15.   
  16. class Player {  
  17.     private int age = 12;  
  18.   
  19.     private Player() {  
  20.         this.age = 50;  
  21.     }  
  22.   
  23.     public int getAge() {  
  24.         return this.age;  
  25.     }  
  26.   
  27.     public void setAge(int age) {  
  28.         this.age = age;  
  29.     }  
  30. }  

2)、使用直接获取内存的方式实现浅克隆

    如何实现浅克隆?在clone(){...}方法中调用super.clone(),对吗?这里存在的问题是首先你必须继续Cloneable接口,并且在所有你需要做浅克隆的对象中实现clone()方法,对于一个懒懒的程序员来说,这个工作量太大了。

    我不推荐上面的做法而是直接使用Unsafe,我们可以仅使用几行代码就实现浅克隆,并且它可以像某些工具类一样用于任意类的克隆。

    这个戏法就是把一个对象的字节码拷贝到内存的另外一个地方,然后再将这个对象转换为被克隆的对象类型。

  3)、来自黑客的密码安全

这个好似很有趣吧?实事就是这样的。开发人员创建密码或者是保证密码到字符串中,然后在应用程序的代码中使用这些密码,使用过后,聪明的程序员会把字符串的引用设为NULL,因此它就不会被引用着并且很容易被垃圾收集器给回收掉。

    但是从你将引用设为NULL到被垃圾收集器收集的这个时间段之内(原文:But from the time, you made the reference null to the time garbage collector kicks in),它是处于字符串池中的,并且在你系统中进行一个复杂的攻击(原文:And a sophisticated attack on your system),也是可以读取到你的内存区域并且获得密码,虽然机会很小,但是总是存在的。

    这就是为什么建议使用char[]数组存放密码,当使用完过后,你可以迭代处理当前数组,修改/清空这些字符。

    另外一个方式就是使用魔术类Unsafe。你可以创建另外一个和当前密码字符串具有相同长度的临时字符串,将临时密码中的每个字符都设值为"?"或者"*"(任何字符都可以),当你完成密码的逻辑后,你只需要简单的将临时密码中的字节数组拷贝到原始的密码串中,这就是使用临时密码覆盖真实的密码。

    示例代码可能会是这样:

  1. String password = new String("l00k@myHor$e");  
  2. String fake = new String(password.replaceAll(".""?"));  
  3. System.out.println(password); // l00k@myHor$e  
  4. System.out.println(fake); // ????????????  
  5.    
  6. getUnsafe().copyMemory(fake, 0L, null, toAddress(password), sizeOf(password));  
  7.    
  8. System.out.println(password); // ????????????  
  9. System.out.println(fake); // ????????????  
 运行时动态创建类

    我们可以在运行时运态的创建类,例如通过编译后的.class文件,操作方式就是将.class文件读取到字节数据组中,并将其传到defineClass方法中。

  1. //Sample code to craeet classes  
  2. byte[] classContents = getClassContent();  
  3. Class c = getUnsafe().defineClass(null, classContents, 0, classContents.length);  
  4. c.getMethod("a").invoke(c.newInstance(), null);  
  5.    
  6. //Method to read .class file  
  7. private static byte[] getClassContent() throws Exception {  
  8.     File f = new File("/home/mishadoff/tmp/A.class");  
  9.     FileInputStream input = new FileInputStream(f);  
  10.     byte[] content = new byte[(int)f.length()];  
  11.     input.read(content);  
  12.     input.close();  
  13.     return content;  
  14. }  

 4)、超大数组

    从所周知,常量Integer.MAX_VALUE是JAVA中数组长度的最大值,如果你想创建一个非常大的数组(虽然在通常的应用中不可能会用上),可以通过对内存进行直接分配实现。

    下面这个示例将会创建分配一段连续的内存(数组),它的容易是允许最大容量的两倍。

  1. class SuperArray {  
  2.     private final static int BYTE = 1;  
  3.     private long size;  
  4.     private long address;  
  5.        
  6.     public SuperArray(long size) {  
  7.         this.size = size;  
  8.         //得到分配内存的起始地址  
  9.         address = getUnsafe().allocateMemory(size * BYTE);  
  10.     }  
  11.     public void set(long i, byte value) {  
  12.         getUnsafe().putByte(address + i * BYTE, value);  
  13.     }  
  14.     public int get(long idx) {  
  15.         return getUnsafe().getByte(address + idx * BYTE);  
  16.     }  
  17.     public long size() {  
  18.         return size;  
  19.     }  
  20. }  
    应用示例

[java]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. long SUPER_SIZE = (long)Integer.MAX_VALUE * 2;  
  2. SuperArray array = new SuperArray(SUPER_SIZE);  
  3. System.out.println("Array size:" + array.size()); // 4294967294  
  4. for (int i = 0; i < 100; i++) {  
  5.     array.set((long)Integer.MAX_VALUE + i, (byte)3);  
  6.     sum += array.get((long)Integer.MAX_VALUE + i);  
  7. }  
  8. System.out.println("Sum of 100 elements:" + sum);  // 300  

    但请注意这可能会导致JVM挂掉。

    

    结束语

    sun.misc.Unsafe provides almost unlimited capabilities for exploring and modification of VM’s runtime data structures. Despite the fact that these capabilities are almost inapplicable in Java development itself, Unsafe is a great tool for anyone who want to study HotSpot VM without C++ code debugging or need to create ad hoc profiling instruments.

    sun.misc.Unsafe提供了可以随意查看及修改JVM中运行时的数据结构,尽管这些功能在JAVA开发本身是不适用的,Unsafe是一个用于研究学习HotSpot虚拟机非常棒的工具,因为它不需要调用C++代码,或者需要创建即时分析的工具。


参考连接:http://blog.csdn.net/fenglibing/article/details/17138079



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值