[关闭]
@chenxuxiong 2016-05-28T06:44:32.000000Z 字数 4202 阅读 799

Java反射,反射性能提高

未分类


1.Java反射机制定义

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

java.lang.Class;
java.lang.reflect.Constructor;
java.lang.reflect.Field;
java.lang.reflect.Method;
java.lang.reflect.Modifier;

2.获得一个类的类对象有哪些方式?

3.如何通过反射创建对象?

4.如何通过反射获取和设置对象私有字段的值?

答:可以通过类对象的getDeclaredField()方法得到字段(Field)对象,然后再通过字段对象的setAccessible(true)将其设置为可以访问,接下来就可以通过get/set方法来获取/设置字段的值了。

5.哪里用到反射机制?

jdbc中有一行代码:Class.forName('com.mysql.jdbc.Driver.class').newInstance();那个时候只知道生成驱动对象实例,后来才知道,这就是反射,现在

很多框架都用到反射机制,hibernate,struts都是用反射机制实现的。
拦截器是基于 java 反射机制的

6.反射机制的优缺点?

但反射有两个缺点。
第一个是性能问题。用于字段和方法接入时反射要远 慢于直接代码。性能问题的程度取决于程序中是如何使用反射的。如果它作为程序运行中相对很少涉及的部分,缓慢的性能将不会是一个问题。即使测试中最坏情况 下的计时图显示的反射操作只耗用几微秒。仅反射在性能关键的应用的核心逻辑中使用时性能问题才变得至关重要。

7.如何通过反射调用对象的方法?

答:先获取类对象getClass(),再通过方法名获取所要方法getMethod("xxx"),最后根据所得方法用invoke(参数)方法执行。

  1. import java.lang.reflect.Method;
  2. class MethodInvokeTest {
  3. public static void main(String[] args) throws Exception {
  4. String str = "hello";
  5. Method m = str.getClass().getMethod("toUpperCase");
  6. System.out.println(m.invoke(str)); // HELLO
  7. }
  8. }

8.原理

Class.forName(classname)方法,实际上是调用了Class类中的 Class.forName(classname, true, currentLoader)方法。参数:name - 所需类的完全限定名;initialize - 是否必须初始化类;loader - 用于加载类的类加载器。currentLoader则是通过调用ClassLoader.getCallerClassLoader()获取当前类加载器的。类要想使用,必须用类加载器加载,所以需要加载器。反射机制,不是每次都去重新反射,而是提供了cache,每次都会需要类加载器去自己的cache中查找,如果可以查到,则直接返回该类。

9.如何提高性能?

1.在系统启动阶段使用反射。
2.尽量不要getMethods()后再遍历筛选,而直接用getMethod(methodName)来根据方法名获取方法
3.缓存:比如,需要多次动态创建一个类的实例的时候,有缓存的写法会比没有缓存要快很多,需要注意线程安全问题
4.将反射得到元数据保存起来,使用时,只需从内存中调用即可。
5.hotspot虚拟机会对执行次数较多的方法进行优化(例如使用jit技术)。
6.使用高性能的反射库,应该会比自己写缓存效果好

  1. import java.lang.reflect.Array;
  2. import java.lang.reflect.Constructor;
  3. import java.lang.reflect.Field;
  4. import java.lang.reflect.Method;
  5. /**
  6. * Java Reflection Cookbook
  7. *
  8. * @author Michael Lee
  9. * @since 2006-8-23
  10. * @version 0.1a
  11. */
  12. public class Reflection {
  13. /**
  14. * 得到某个对象的公共属性
  15. *
  16. * @param owner, fieldName
  17. * @return 该属性对象
  18. * @throws Exception
  19. *
  20. */
  21. public Object getProperty(Object owner, String fieldName) throws Exception {
  22. Class ownerClass = owner.getClass();
  23. Field field = ownerClass.getField(fieldName);
  24. Object property = field.get(owner);
  25. return property;
  26. }
  27. /**
  28. * 得到某类的静态公共属性
  29. *
  30. * @param className 类名
  31. * @param fieldName 属性名
  32. * @return 该属性对象
  33. * @throws Exception
  34. */
  35. public Object getStaticProperty(String className, String fieldName)
  36. throws Exception {
  37. Class ownerClass = Class.forName(className);
  38. Field field = ownerClass.getField(fieldName);
  39. Object property = field.get(ownerClass);
  40. return property;
  41. }
  42. /**
  43. * 执行某对象方法
  44. *
  45. * @param owner
  46. * 对象
  47. * @param methodName
  48. * 方法名
  49. * @param args
  50. * 参数
  51. * @return 方法返回值
  52. * @throws Exception
  53. */
  54. public Object invokeMethod(Object owner, String methodName, Object[] args)
  55. throws Exception {
  56. Class ownerClass = owner.getClass();
  57. Class[] argsClass = new Class[args.length];
  58. for (int i = 0, j = args.length; i < j; i++) {
  59. argsClass[i] = args[i].getClass();
  60. }
  61. Method method = ownerClass.getMethod(methodName, argsClass);
  62. return method.invoke(owner, args);
  63. }
  64. /**
  65. * 执行某类的静态方法
  66. *
  67. * @param className
  68. * 类名
  69. * @param methodName
  70. * 方法名
  71. * @param args
  72. * 参数数组
  73. * @return 执行方法返回的结果
  74. * @throws Exception
  75. */
  76. public Object invokeStaticMethod(String className, String methodName,
  77. Object[] args) throws Exception {
  78. Class ownerClass = Class.forName(className);
  79. Class[] argsClass = new Class[args.length];
  80. for (int i = 0, j = args.length; i < j; i++) {
  81. argsClass[i] = args[i].getClass();
  82. }
  83. Method method = ownerClass.getMethod(methodName, argsClass);
  84. return method.invoke(null, args);
  85. }
  86. /**
  87. * 新建实例
  88. *
  89. * @param className
  90. * 类名
  91. * @param args
  92. * 构造函数的参数
  93. * @return 新建的实例
  94. * @throws Exception
  95. */
  96. public Object newInstance(String className, Object[] args) throws Exception {
  97. Class newoneClass = Class.forName(className);
  98. Class[] argsClass = new Class[args.length];
  99. for (int i = 0, j = args.length; i < j; i++) {
  100. argsClass[i] = args[i].getClass();
  101. }
  102. Constructor cons = newoneClass.getConstructor(argsClass);
  103. return cons.newInstance(args);
  104. }
  105. /**
  106. * 是不是某个类的实例
  107. * @param obj 实例
  108. * @param cls 类
  109. * @return 如果 obj 是此类的实例,则返回 true
  110. */
  111. public boolean isInstance(Object obj, Class cls) {
  112. return cls.isInstance(obj);
  113. }
  114. /**
  115. * 得到数组中的某个元素
  116. * @param array 数组
  117. * @param index 索引
  118. * @return 返回指定数组对象中索引组件的值
  119. */
  120. public Object getByArray(Object array, int index) {
  121. return Array.get(array,index);
  122. }
  123. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注