写一些Java框架的时候,经常需要通过反射get或者set某个bean的field,比较普通的做法是获取field后调用java.lang.reflect.Field.get(Object),但每次都这样调用,能否有优化的空间呢?

答案是有。

第一种:

由于每次都是重复的调用,所以想到了缓存每个bean的field,但这样做还是不够,所以想到了写一个code generator。通过生成代码的方式,get或者set每个bean的时候直接调用该bean的getter或者setter,这个实现听起来很牛逼,其实就是用asm生成一个类在用一个classloader加载进来每次调用直接invoke就可以了。

可单纯为了一个反射调用做这么多,总感觉是大炮打了蚊子。

第二种:

多谢@RednaxelaFX   的指点,找到了更简单的做法:sun.misc.Unsafe

使用也非常的简单:首先通过sun.misc.Unsafe.objectFieldOffset(Field) 获取field的offset,然后使用sun.misc.Unsafe.getObject(Object, long)获取某个实例上的field的值。

简单的测试代码如下:

  1. import java.io.Serializable;
  2. import java.lang.reflect.Field;
  3. import sun.misc.Unsafe;
  4. /**
  5. * @author haitao.yao Dec 14, 2010
  6. */
  7. public class ReflectionCompare {
  8. private static final int count = 10000000;
  9. /**
  10. * @param args
  11. */
  12. public static void main(String[] args) {
  13. long duration = testIntCommon();
  14. System.out.println("int common test for  " + count
  15. + " times, duration: " + duration);
  16. duration = testUnsafe();
  17. System.out.println("int unsafe test for  " + count
  18. + " times, duration: " + duration);
  19. }
  20. private static long testUnsafe() {
  21. long start = System.currentTimeMillis();
  22. sun.misc.Unsafe unsafe = getUnsafe();
  23. int temp = count;
  24. Field field = getIntField();
  25. long offset = unsafe.objectFieldOffset(field);
  26. while (temp-- > 0) {
  27. unsafe.getInt(new TestBean(), offset);
  28. }
  29. return System.currentTimeMillis() - start;
  30. }
  31. private static long testIntCommon() {
  32. long start = System.currentTimeMillis();
  33. int temp = count;
  34. getIntField().setAccessible(true);
  35. while (temp-- > 0) {
  36. TestBean bean = new TestBean();
  37. try {
  38. getIntField().get(bean);
  39. } catch (Exception e) {
  40. e.printStackTrace();
  41. }
  42. }
  43. return System.currentTimeMillis() - start;
  44. }
  45. private static final sun.misc.Unsafe unsafe;
  46. static {
  47. sun.misc.Unsafe value = null;
  48. try {
  49. Class<?> clazz = Class.forName("sun.misc.Unsafe");
  50. Field field = clazz.getDeclaredField("theUnsafe");
  51. field.setAccessible(true);
  52. value = (Unsafe) field.get(null);
  53. } catch (Exception e) {
  54. e.printStackTrace();
  55. throw new RuntimeException("error to get theUnsafe", e);
  56. }
  57. unsafe = value;
  58. }
  59. public static final sun.misc.Unsafe getUnsafe() {
  60. return unsafe;
  61. }
  62. private static final Field intField;
  63. private static final Field stringField;
  64. static {
  65. try {
  66. intField = TestBean.class.getDeclaredField("age");
  67. stringField = TestBean.class.getDeclaredField("name");
  68. } catch (Exception e) {
  69. e.printStackTrace();
  70. throw new IllegalStateException("failed to init testbean field", e);
  71. }
  72. }
  73. public static final Field getIntField() {
  74. return intField;
  75. }
  76. public static final Field getStringField() {
  77. return stringField;
  78. }
  79. /**
  80. * @author haitao.yao
  81. * Dec 14, 2010
  82. */
  83. static class TestBean implements Serializable{
  84. /**
  85. *
  86. */
  87. private static final long serialVersionUID = -5994966479456252766L;
  88. private String name;
  89. private int age;
  90. /**
  91. * @return the name
  92. */
  93. public String getName() {
  94. return name;
  95. }
  96. /**
  97. * @param name the name to set
  98. */
  99. public void setName(String name) {
  100. this.name = name;
  101. }
  102. /**
  103. * @return the age
  104. */
  105. public int getAge() {
  106. return age;
  107. }
  108. /**
  109. * @param age the age to set
  110. */
  111. public void setAge(int age) {
  112. this.age = age;
  113. }
  114. }
  115. }

import java.io.Serializable;
import java.lang.reflect.Field;
import sun.misc.Unsafe;
/**
* @author haitao.yao Dec 14, 2010
*/
public class ReflectionCompare {
private static final int count = 10000000;
/**
* @param args
*/
public static void main(String[] args) {
long duration = testIntCommon();
System.out.println("int common test for " + count
+ " times, duration: " + duration);
duration = testUnsafe();
System.out.println("int unsafe test for " + count
+ " times, duration: " + duration);
}
private static long testUnsafe() {
long start = System.currentTimeMillis();
sun.misc.Unsafe unsafe = getUnsafe();
int temp = count;
Field field = getIntField();
long offset = unsafe.objectFieldOffset(field);
while (temp-- > 0) {
unsafe.getInt(new TestBean(), offset);
}
return System.currentTimeMillis() - start;
}
private static long testIntCommon() {
long start = System.currentTimeMillis();
int temp = count;
getIntField().setAccessible(true);
while (temp-- > 0) {
TestBean bean = new TestBean();
try {
getIntField().get(bean);
} catch (Exception e) {
e.printStackTrace();
}
}
return System.currentTimeMillis() - start;
}
private static final sun.misc.Unsafe unsafe;
static {
sun.misc.Unsafe value = null;
try {
Class<?> clazz = Class.forName("sun.misc.Unsafe");
Field field = clazz.getDeclaredField("theUnsafe");
field.setAccessible(true);
value = (Unsafe) field.get(null);
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException("error to get theUnsafe", e);
}
unsafe = value;
}
public static final sun.misc.Unsafe getUnsafe() {
return unsafe;
}
private static final Field intField;
private static final Field stringField;
static {
try {
intField = TestBean.class.getDeclaredField("age");
stringField = TestBean.class.getDeclaredField("name");
} catch (Exception e) {
e.printStackTrace();
throw new IllegalStateException("failed to init testbean field", e);
}
}
public static final Field getIntField() {
return intField;
}
public static final Field getStringField() {
return stringField;
}

/**
* @author haitao.yao
* Dec 14, 2010
*/
static class TestBean implements Serializable{
/**
*
*/
private static final long serialVersionUID = -5994966479456252766L;

private String name;
private int age;
/**
* @return the name
*/
public String getName() {
return name;
}
/**
* @param name the name to set
*/
public void setName(String name) {
this.name = name;
}
/**
* @return the age
*/
public int getAge() {
return age;
}
/**
* @param age the age to set
*/
public void setAge(int age) {
this.age = age;
}
}
}

通过测试发现,效率是普通java.lang.reflect.Field.get(Object)的3倍,当然,性能这个东西,还是自己测试了放心。

这样做有一个不好的地方:sun.misc.Unsafe在sun的包里,默认情况下,eclipse编译会报错,在Window->Preference->Java->Compiler->Errors/Warnings->Deprecated and restricted API -> Forbidden Reference 修改成Warning或者Ignore就可以了。由于Unsafe在JDK中很多的类库中都在使用,框架代码中使用还是很安全的,如果需要api变动,JDK源代码的修改工作量比我们的大多了 :-0

至于第一种方法,虽然麻烦,有时间还是可以尝试一下的,有时间了写一下。

Java 反射调用的一种优化的更多相关文章

  1. Java 反射 调用私有域和方法(setAccessible)

    Java 反射 调用私有域和方法(setAccessible) @author ixenos AccessibleObject类 Method.Field和Constructor类共同继承了Acces ...

  2. 利用java反射调用类的的私有方法--转

    原文:http://blog.csdn.net/woshinia/article/details/11766567 1,今天和一位朋友谈到父类私有方法的调用问题,本来以为利用反射很轻松就可以实现,因为 ...

  3. JAVA反射调用方法

    1.用户类 package com.lf.entity; import com.lf.annotation.SetProperty; import com.lf.annotation.SetTable ...

  4. 通过Java反射调用方法

    这是个测试用的例子,通过反射调用对象的方法.     TestRef.java import java.lang.reflect.Method; import java.lang.reflect.In ...

  5. java反射调用dubbo接口

    需求:项目增加幂等 场景:1.三个项目:a .b.c2.a项目加幂等3.b项目dubbo调用项目a的时候超时没有获取返回结果,增加重试机制(非立即重试,3min or 5min 后重试)4.c项目是一 ...

  6. Java 反射 调用私有构造方法

    单例类: package singleton; public class SingletonTest { // 私有构造方法 private SingletonTest(){ System.out.p ...

  7. java黑魔法-反射机制-02-通过Java反射调用其他类方法

    package com.aaron.reflect; import java.lang.reflect.Method; import java.lang.reflect.InvocationTarge ...

  8. java反射调用api

    cglib的fastmethod 简单示例: FastClass serviceFastClass = FastClass.create(Person.class); Person p = new P ...

  9. Java 反射调用方法 - 不跳过安全检查、跳过安全检查和普通方法性能比较测试

    java中反射提供灵活性同时,给运行效率带来了一定影响.写个代码测试一下 package com.xzlf.reflectTest; import java.lang.reflect.Method; ...

随机推荐

  1. BZOJ1293 [SCOI2009]生日礼物 【队列】

    题目 小西有一条很长的彩带,彩带上挂着各式各样的彩珠.已知彩珠有N个,分为K种.简单的说,可以将彩带考虑为x轴,每一个彩珠有一个对应的坐标(即位置).某些坐标上可以没有彩珠,但多个彩珠也可以出现在同一 ...

  2. BZOJ3132 上帝造题的七分钟 【二维树状数组】

    题目 "第一分钟,X说,要有矩阵,于是便有了一个里面写满了0的n×m矩阵. 第二分钟,L说,要能修改,于是便有了将左上角为(a,b),右下角为(c,d)的一个矩形区域内的全部数字加上一个值的 ...

  3. Android中代码设置RadioButton的高端技巧

    不知道怎么起标题,就这样了. 目前主要讲两个方面内容: 代码方式 设置RadioButton的 android:button . android:background 等属性为 @null : 代码方 ...

  4. C# Async await和Task的关系

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  5. POJ2240:Arbitrage(最长路+正环)

    Arbitrage Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 29374   Accepted: 12279 题目链接: ...

  6. HDU2389:Rain on your Parade(二分图最大匹配+HK算法)

    Rain on your Parade Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 655350/165535 K (Java/Ot ...

  7. ICE学习笔记一----运行官方的java版demo程序

    建议新手和我一样,从官网下载英文文档,开个有道词典,慢慢啃. 官方文档下载: http://download.csdn.net/detail/xiong_mao_1/6300631 程序代码就不说了, ...

  8. Python爬虫学习笔记之抓取猫眼的排行榜

    代码: import json import requests from requests.exceptions import RequestException import re import ti ...

  9. 慕课网javascript 进阶篇 第九章 编程练习

    把平常撸的码来博客上再撸一遍既可以加深理解,又可以理清思维.还是很纯很纯的小白,各位看官老爷们,不要嫌弃.最近都是晚睡,昨晚也不例外,两点多睡的.故,八点起来的人不是很舒服,脑袋有点晕呼呼,鉴于昨晚看 ...

  10. 解决Idea Jsp <%%>中 request resopnse等无自动提示的问题

    解决办法:缺少Apache的lib依赖, 只需 File->Project Srructure->Libraries 加号 找到Apache安装的lib目录添加依赖即可.亲测可用