1. 概述

定义

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

用途

在日常的第三方应用开发过程中,经常会遇到某个类的某个成员变量、方法或是属性是私有的或是只对系统应用开放,这时候就可以利用 Java 的反射机制通过反射来获取所需的私有成员或是方法。当然,也不是所有的都适合反射,之前就遇到一个案例,通过反射得到的结果与预期不符。阅读源码发现,经过层层调用后在最终返回结果的地方对应用的权限进行了校验,对于没有权限的应用返回值是没有意义的缺省值,否则返回实际值起到保护用户的隐私目的。

2. 反射机制的相关类

与 Java 反射相关的类如下:

类名 用途
Class 类 代表类的实体,在运行的 Java 应用程序中表示类和接口
Field 类 代表类的成员变量(成员变量也称为类的属性)
Method 类 代表类的方法
Constructor 类 代表类的构造方法

Class 类

Class 代表类的实体,在运行的 Java 应用程序中表示类和接口。在这个类中提供了很多有用的方法,这里对他们简单的分类介绍。

  • 获得类相关的方法 | 方法 | 用途 | | —- | —- | | asSubclass(Class clazz) | 把传递的类的对象转换成代表其子类的对象 | | Cast | 把对象转换成代表类或是接口的对象 | | getClassLoader() | 获得类的加载器 | | getClasses() | 返回一个数组,数组中包含该类中所有公共类和接口类的对象 | | getDeclaredClasses() | 返回一个数组,数组中包含该类中所有类和接口类的对象 | | forName(String className) | 根据类名返回类的对象 | | getName() | 获得类的完整路径名字 | | newInstance() | 创建类的实例 | | getPackage() | 获得类的包 | | getSimpleName() | 获得类的名字 | | getSuperclass() | 获得当前类继承的父类的名字 | | getInterfaces() | 获得当前类实现的类或是接口 |
  • 获得类中属性相关的方法 | 方法 | 用途 | | —- | —- | | getField(String name) | 获得某个公有的属性对象 | | getFields() | 获得所有公有的属性对象 | | getDeclaredField(String name) | 获得某个属性对象 | | getDeclaredFields() | 获得所有属性对象 |
  • 获得类中构造器相关的方法 | 方法 | 用途 | | —- | —- | | getConstructor(Class…<?> parameterTypes) | 获得该类中与参数类型匹配的公有构造方法 | | getConstructors() | 获得该类的所有公有构造方法 | | getDeclaredConstructor(Class…<?> parameterTypes) | 获得该类中与参数类型匹配的构造方法 | | getDeclaredConstructors() | 获得该类所有构造方法 |

  • 获得类中方法相关的方法 | 方法 | 用途 | | —- | —- | | getMethod(String name, Class…<?> parameterTypes) | 获得该类某个公有的方法 | | getMethods() | 获得该类所有公有的方法 | | getDeclaredMethod(String name, Class…<?> parameterTypes) | 获得该类某个方法 | | getDeclaredMethods() | 获得该类所有方法 |

  • 类中其他重要的方法 | 方法 | 用途 | | —- | —- | | isAnnotation() | 如果是注解类型则返回 true | | isAnnotationPresent(Class<? extends Annotation> annotationClass) | 如果是指定类型注解类型则返回 true | | isAnonymousClass() | 如果是匿名类则返回 true | | isArray() | 如果是一个数组类则返回 true | | isEnum() | 如果是枚举类则返回 true | | isInstance(Object obj) | 如果 obj 是该类的实例则返回 true | | isInterface() | 如果是接口类则返回 true | | isLocalClass() | 如果是局部类则返回 true | | isMemberClass() | 如果是内部类则返回 true |

Field 类

Field 代表类的成员变量(成员变量也称为类的属性)。

方法 用途
equals(Object obj) 属性与 obj 相等则返回 true
get(Object obj) 获得 obj 中对应的属性值
set(Object obj, Object value) 设置 obj 中对应属性值

Method 类

Method 代表类的方法。

方法 用途
invoke(Object obj, Object… args) 传递 object 对象及参数调用该对象对应的方法

Constructor 类

Constructor 代表类的构造方法。

方法 用途
newInstance(Object… initargs) 根据传递的参数创建类的对象

3. 示例

为了演示反射的使用,首先构造一个与书籍相关的 model——Book.java,然后通过反射方法示例创建对象、反射私有构造方法、反射私有属性、反射私有方法,最后给出两个比较复杂的反射示例——获得当前 ZenMode 和关机 Shutdown。

  • 被反射类 Book.java

    1. public class Book{
    2. private final static String TAG = "BookTag";
    3. private String name;
    4. private String author;
    5. @Override
    6. public String toString() {
    7. return "Book{" +
    8. "name='" + name + '\'' +
    9. ", author='" + author + '\'' +
    10. '}';
    11. }
    12. public Book() {
    13. }
    14. private Book(String name, String author) {
    15. this.name = name;
    16. this.author = author;
    17. }
    18. public String getName() {
    19. return name;
    20. }
    21. public void setName(String name) {
    22. this.name = name;
    23. }
    24. public String getAuthor() {
    25. return author;
    26. }
    27. public void setAuthor(String author) {
    28. this.author = author;
    29. }
    30. private String declaredMethod(int index) {
    31. String string = null;
    32. switch (index) {
    33. case 0:
    34. string = "I am declaredMethod 1 !";
    35. break;
    36. case 1:
    37. string = "I am declaredMethod 2 !";
    38. break;
    39. default:
    40. string = "I am declaredMethod 1 !";
    41. }
    42. return string;
    43. }
    44. }
  • 反射逻辑封装在 ReflectClass.java

    1. public class ReflectClass {
    2. private final static String TAG = "peter.log.ReflectClass";
    3. // 创建对象
    4. public static void reflectNewInstance() {
    5. try {
    6. Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
    7. Object objectBook = classBook.newInstance();
    8. Book book = (Book) objectBook;
    9. book.setName("Android进阶之光");
    10. book.setAuthor("刘望舒");
    11. Log.d(TAG,"reflectNewInstance book = " + book.toString());
    12. } catch (Exception ex) {
    13. ex.printStackTrace();
    14. }
    15. }
    16. // 反射私有的构造方法
    17. public static void reflectPrivateConstructor() {
    18. try {
    19. Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
    20. Constructor<?> declaredConstructorBook = classBook.getDeclaredConstructor(String.class,String.class);
    21. declaredConstructorBook.setAccessible(true);
    22. Object objectBook = declaredConstructorBook.newInstance("Android开发艺术探索","任玉刚");
    23. Book book = (Book) objectBook;
    24. Log.d(TAG,"reflectPrivateConstructor book = " + book.toString());
    25. } catch (Exception ex) {
    26. ex.printStackTrace();
    27. }
    28. }
    29. // 反射私有属性
    30. public static void reflectPrivateField() {
    31. try {
    32. Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
    33. Object objectBook = classBook.newInstance();
    34. Field fieldTag = classBook.getDeclaredField("TAG");
    35. fieldTag.setAccessible(true);
    36. String tag = (String) fieldTag.get(objectBook);
    37. Log.d(TAG,"reflectPrivateField tag = " + tag);
    38. } catch (Exception ex) {
    39. ex.printStackTrace();
    40. }
    41. }
    42. // 反射私有方法
    43. public static void reflectPrivateMethod() {
    44. try {
    45. Class<?> classBook = Class.forName("com.android.peter.reflectdemo.Book");
    46. Method methodBook = classBook.getDeclaredMethod("declaredMethod",int.class);
    47. methodBook.setAccessible(true);
    48. Object objectBook = classBook.newInstance();
    49. String string = (String) methodBook.invoke(objectBook,0);
    50. Log.d(TAG,"reflectPrivateMethod string = " + string);
    51. } catch (Exception ex) {
    52. ex.printStackTrace();
    53. }
    54. }
    55. // 获得系统Zenmode值
    56. public static int getZenMode() {
    57. int zenMode = -1;
    58. try {
    59. Class<?> cServiceManager = Class.forName("android.os.ServiceManager");
    60. Method mGetService = cServiceManager.getMethod("getService", String.class);
    61. Object oNotificationManagerService = mGetService.invoke(null, Context.NOTIFICATION_SERVICE);
    62. Class<?> cINotificationManagerStub = Class.forName("android.app.INotificationManager$Stub");
    63. Method mAsInterface = cINotificationManagerStub.getMethod("asInterface",IBinder.class);
    64. Object oINotificationManager = mAsInterface.invoke(null,oNotificationManagerService);
    65. Method mGetZenMode = cINotificationManagerStub.getMethod("getZenMode");
    66. zenMode = (int) mGetZenMode.invoke(oINotificationManager);
    67. } catch (Exception ex) {
    68. ex.printStackTrace();
    69. }
    70. return zenMode;
    71. }
    72. // 关闭手机
    73. public static void shutDown() {
    74. try {
    75. Class<?> cServiceManager = Class.forName("android.os.ServiceManager");
    76. Method mGetService = cServiceManager.getMethod("getService",String.class);
    77. Object oPowerManagerService = mGetService.invoke(null,Context.POWER_SERVICE);
    78. Class<?> cIPowerManagerStub = Class.forName("android.os.IPowerManager$Stub");
    79. Method mShutdown = cIPowerManagerStub.getMethod("shutdown",boolean.class,String.class,boolean.class);
    80. Method mAsInterface = cIPowerManagerStub.getMethod("asInterface",IBinder.class);
    81. Object oIPowerManager = mAsInterface.invoke(null,oPowerManagerService);
    82. mShutdown.invoke(oIPowerManager,true,null,true);
    83. } catch (Exception ex) {
    84. ex.printStackTrace();
    85. }
    86. }
    87. public static void shutdownOrReboot(final boolean shutdown, final boolean confirm) {
    88. try {
    89. Class<?> ServiceManager = Class.forName("android.os.ServiceManager");
    90. // 获得ServiceManager的getService方法
    91. Method getService = ServiceManager.getMethod("getService", java.lang.String.class);
    92. // 调用getService获取RemoteService
    93. Object oRemoteService = getService.invoke(null, Context.POWER_SERVICE);
    94. // 获得IPowerManager.Stub类
    95. Class<?> cStub = Class.forName("android.os.IPowerManager$Stub");
    96. // 获得asInterface方法
    97. Method asInterface = cStub.getMethod("asInterface", android.os.IBinder.class);
    98. // 调用asInterface方法获取IPowerManager对象
    99. Object oIPowerManager = asInterface.invoke(null, oRemoteService);
    100. if (shutdown) {
    101. // 获得shutdown()方法
    102. Method shutdownMethod = oIPowerManager.getClass().getMethod(
    103. "shutdown", boolean.class, String.class, boolean.class);
    104. // 调用shutdown()方法
    105. shutdownMethod.invoke(oIPowerManager, confirm, null, false);
    106. } else {
    107. // 获得reboot()方法
    108. Method rebootMethod = oIPowerManager.getClass().getMethod("reboot",
    109. boolean.class, String.class, boolean.class);
    110. // 调用reboot()方法
    111. rebootMethod.invoke(oIPowerManager, confirm, null, false);
    112. }
    113. } catch (Exception e) {
    114. e.printStackTrace();
    115. }
    116. }
    117. }
  • 调用相应反射逻辑方法

    1. try {
    2. // 创建对象
    3. ReflectClass.reflectNewInstance();
    4. // 反射私有的构造方法
    5. ReflectClass.reflectPrivateConstructor();
    6. // 反射私有属性
    7. ReflectClass.reflectPrivateField();
    8. // 反射私有方法
    9. ReflectClass.reflectPrivateMethod();
    10. } catch (Exception ex) {
    11. ex.printStackTrace();
    12. }
    13. Log.d(TAG," zenmode = " + ReflectClass.getZenMode());
    1. <br />Log 输出结果如下:
    1. 08-27 15:11:37.999 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectNewInstance book = Book{name='Android进阶之光', author='刘望舒'}
    2. 08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateConstructor book = Book{name='Android开发艺术探索', author='任玉刚'}
    3. 08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateField tag = BookTag
    4. 08-27 15:11:38.000 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectClass: reflectPrivateMethod string = I am declaredMethod 1 !
    5. 08-27 15:11:38.004 11987-11987/com.android.peter.reflectdemo D/peter.log.ReflectDemo: zenmode = 0

    4. 总结

    本文列举了反射机制使用过程中常用的、重要的一些类及其方法,更多信息和用法需要近一步的阅读 Google 提供的相关文档和示例。

在阅读 Class 类文档时发现一个特点,以通过反射获得 Method 对象为例,一般会提供四种方法,getMethod(parameterTypes)、getMethods()、getDeclaredMethod(parameterTypes) 和 getDeclaredMethods()。getMethod(parameterTypes) 用来获取某个公有的方法的对象,getMethods() 获得该类所有公有的方法,getDeclaredMethod(parameterTypes) 获得该类某个方法,getDeclaredMethods() 获得该类所有方法。带有 Declared 修饰的方法可以反射到私有的方法,没有 Declared 修饰的只能用来反射公有的方法。其他的 Annotation、Field、Constructor 也是如此。

在 ReflectClass 类中还提供了两种反射 PowerManager.shutdown() 的方法,在调用的时候会输出如下 log,提示没有相关权限。之前在项目中尝试反射其他方法的时候还遇到过有权限和没权限返回的值不一样的情况。如果源码中明确进行了权限验证,而你的应用又无法获得这个权限的话,建议就不要浪费时间反射了。

  1. W/System.err: java.lang.reflect.InvocationTargetException
  2. W/System.err: at java.lang.reflect.Method.invoke(Native Method)
  3. W/System.err: at .ReflectClass.shutDown(ReflectClass.java:104)
  4. W/System.err: at .MainActivity$1.onClick(MainActivity.java:25)
  5. W/System.err: at android.view.View.performClick(View.java:6259)
  6. W/System.err: at android.view.View$PerformClick.run(View.java:24732)
  7. W/System.err: at android.os.Handler.handleCallback(Handler.java:789)
  8. W/System.err: at android.os.Handler.dispatchMessage(Handler.java:98)
  9. W/System.err: at android.os.Looper.loop(Looper.java:164)
  10. W/System.err: at android.app.ActivityThread.main(ActivityThread.java:6592)
  11. W/System.err: at java.lang.reflect.Method.invoke(Native Method)
  12. W/System.err: at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
  13. W/System.err: at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:769)
  14. W/System.err: Caused by: java.lang.SecurityException: Neither user 10224 nor current process has android.permission.REBOOT.
  15. W/System.err: at android.os.Parcel.readException(Parcel.java:1942)
  16. W/System.err: at android.os.Parcel.readException(Parcel.java:1888)
  17. W/System.err: at android.os.IPowerManager$Stub$Proxy.shutdown(IPowerManager.java:787)
  18. W/System.err: ... 12 more

ReflectDemo

参考文献

认识反射机制(Reflection)
Java 反射机制
一个例子让你了解 Java 反射机制
Java 反射机制的原理及在 Android 下的简单应用
java 中的反射机制
Android 注解与反射机制
java.lang.reflect.Method