反射

反射的概述

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

获取Class类对象的三种方式

  • 三种方式分类
    • 类名.class属性
    • 对象名.getClass()方法
    • Class.forName(全类名)方法

08_获取Class对象的三种方式.png

  • 示例代码

    1. public class Student {
    2. private String name;
    3. private int age;
    4. public Student() {
    5. }
    6. public Student(String name, int age) {
    7. this.name = name;
    8. this.age = age;
    9. }
    10. public String getName() {
    11. return name;
    12. }
    13. public void setName(String name) {
    14. this.name = name;
    15. }
    16. public int getAge() {
    17. return age;
    18. }
    19. public void setAge(int age) {
    20. this.age = age;
    21. }
    22. public void study(){
    23. System.out.println("学生在学习");
    24. }
    25. @Override
    26. public String toString() {
    27. return "Student{" +
    28. "name='" + name + '\'' +
    29. ", age=" + age +
    30. '}';
    31. }
    32. }
    33. public class ReflectDemo1 {
    34. public static void main(String[] args) throws ClassNotFoundException {
    35. //1.Class类中的静态方法forName("全类名")
    36. //全类名:包名 + 类名
    37. Class clazz = Class.forName("com.itheima.myreflect2.Student");
    38. System.out.println(clazz);
    39. //2.通过class属性来获取
    40. Class clazz2 = Student.class;
    41. System.out.println(clazz2);
    42. //3.利用对象的getClass方法来获取class对象
    43. //getClass方法是定义在Object类中.
    44. Student s = new Student();
    45. Class clazz3 = s.getClass();
    46. System.out.println(clazz3);
    47. System.out.println(clazz == clazz2);
    48. System.out.println(clazz2 == clazz3);
    49. }
    50. }

反射获取构造方法并使用

Class类获取构造方法对象的方法

  • 方法介绍
    | 方法名 | 说明 | | —- | —- | | Constructor<?>[] getConstructors() | 返回所有公共构造方法对象的数组 | | Constructor<?>[] getDeclaredConstructors() | 返回所有构造方法对象的数组 | | Constructor getConstructor(Class<?>… parameterTypes) | 返回单个公共构造方法对象 | | Constructor getDeclaredConstructor(Class<?>… parameterTypes) | 返回单个构造方法对象 |

  • 示例代码

    1. public class Student {
    2. private String name;
    3. private int age;
    4. //私有的有参构造方法
    5. private Student(String name) {
    6. System.out.println("name的值为:" + name);
    7. System.out.println("private...Student...有参构造方法");
    8. }
    9. //公共的无参构造方法
    10. public Student() {
    11. System.out.println("public...Student...无参构造方法");
    12. }
    13. //公共的有参构造方法
    14. public Student(String name, int age) {
    15. System.out.println("name的值为:" + name + "age的值为:" + age);
    16. System.out.println("public...Student...有参构造方法");
    17. }
    18. }
    19. public class ReflectDemo1 {
    20. public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
    21. //method1();
    22. //method2();
    23. //method3();
    24. //method4();
    25. }
    26. private static void method4() throws ClassNotFoundException, NoSuchMethodException {
    27. // Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes):
    28. // 返回单个构造方法对象
    29. //1.获取Class对象
    30. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    31. Constructor constructor = clazz.getDeclaredConstructor(String.class);
    32. System.out.println(constructor);
    33. }
    34. private static void method3() throws ClassNotFoundException, NoSuchMethodException {
    35. // Constructor<T> getConstructor(Class<?>... parameterTypes):
    36. // 返回单个公共构造方法对象
    37. //1.获取Class对象
    38. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    39. //小括号中,一定要跟构造方法的形参保持一致.
    40. Constructor constructor1 = clazz.getConstructor();
    41. System.out.println(constructor1);
    42. Constructor constructor2 = clazz.getConstructor(String.class, int.class);
    43. System.out.println(constructor2);
    44. //因为Student类中,没有只有一个int的构造,所以这里会报错.
    45. Constructor constructor3 = clazz.getConstructor(int.class);
    46. System.out.println(constructor3);
    47. }
    48. private static void method2() throws ClassNotFoundException {
    49. // Constructor<?>[] getDeclaredConstructors():
    50. // 返回所有构造方法对象的数组
    51. //1.获取Class对象
    52. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    53. Constructor[] constructors = clazz.getDeclaredConstructors();
    54. for (Constructor constructor : constructors) {
    55. System.out.println(constructor);
    56. }
    57. }
    58. private static void method1() throws ClassNotFoundException {
    59. // Constructor<?>[] getConstructors():
    60. // 返回所有公共构造方法对象的数组
    61. //1.获取Class对象
    62. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    63. Constructor[] constructors = clazz.getConstructors();
    64. for (Constructor constructor : constructors) {
    65. System.out.println(constructor);
    66. }
    67. }
    68. }

Constructor类用于创建对象的方法

  • 方法介绍
    | 方法名 | 说明 | | —- | —- | | T newInstance(Object…initargs) | 根据指定的构造方法创建对象 | | setAccessible(boolean flag) | 设置为true,表示取消访问检查 |

  • 示例代码

    1. // Student类同上一个示例,这里就不在重复提供了
    2. public class ReflectDemo2 {
    3. public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
    4. //T newInstance(Object... initargs):根据指定的构造方法创建对象
    5. //method1();
    6. //method2();
    7. //method3();
    8. //method4();
    9. }
    10. private static void method4() throws ClassNotFoundException, NoSuchMethodException, InstantiationException, IllegalAccessException, InvocationTargetException {
    11. //获取一个私有的构造方法并创建对象
    12. //1.获取class对象
    13. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    14. //2.获取一个私有化的构造方法.
    15. Constructor constructor = clazz.getDeclaredConstructor(String.class);
    16. //被private修饰的成员,不能直接使用的
    17. //如果用反射强行获取并使用,需要临时取消访问检查
    18. constructor.setAccessible(true);
    19. //3.直接创建对象
    20. Student student = (Student) constructor.newInstance("zhangsan");
    21. System.out.println(student);
    22. }
    23. private static void method3() throws ClassNotFoundException, InstantiationException, IllegalAccessException {
    24. //简写格式
    25. //1.获取class对象
    26. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    27. //2.在Class类中,有一个newInstance方法,可以利用空参直接创建一个对象
    28. Student student = (Student) clazz.newInstance();//这个方法现在已经过时了,了解一下
    29. System.out.println(student);
    30. }
    31. private static void method2() throws ClassNotFoundException, NoSuchMethodException, InstantiationException, IllegalAccessException, InvocationTargetException {
    32. //1.获取class对象
    33. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    34. //2.获取构造方法对象
    35. Constructor constructor = clazz.getConstructor();
    36. //3.利用空参来创建Student的对象
    37. Student student = (Student) constructor.newInstance();
    38. System.out.println(student);
    39. }
    40. private static void method1() throws ClassNotFoundException, NoSuchMethodException, InstantiationException, IllegalAccessException, InvocationTargetException {
    41. //1.获取class对象
    42. Class clazz = Class.forName("com.itheima.myreflect3.Student");
    43. //2.获取构造方法对象
    44. Constructor constructor = clazz.getConstructor(String.class, int.class);
    45. //3.利用newInstance创建Student的对象
    46. Student student = (Student) constructor.newInstance("zhangsan", 23);
    47. System.out.println(student);
    48. }
    49. }

小结

  • 获取class对象
    三种方式: Class.forName(“全类名”), 类名.class, 对象名.getClass()
  • 获取里面的构造方法对象
    getConstructor (Class<?>… parameterTypes)
    getDeclaredConstructor (Class<?>… parameterTypes)
  • 如果是public的,直接创建对象
    newInstance(Object… initargs)
  • 如果是非public的,需要临时取消检查,然后再创建对象
    setAccessible(boolean) 暴力反射

反射获取成员变量并使用

Class类获取成员变量对象的方法

  • 方法分类
    | 方法名 | 说明 | | —- | —- | | Field[] getFields() | 返回所有公共成员变量对象的数组 | | Field[] getDeclaredFields() | 返回所有成员变量对象的数组 | | Field getField(String name) | 返回单个公共成员变量对象 | | Field getDeclaredField(String name) | 返回单个成员变量对象 |

  • 示例代码

    1. public class Student {
    2. public String name;
    3. public int age;
    4. public String gender;
    5. private int money = 300;
    6. @Override
    7. public String toString() {
    8. return "Student{" +
    9. "name='" + name + '\'' +
    10. ", age=" + age +
    11. ", gender='" + gender + '\'' +
    12. ", money=" + money +
    13. '}';
    14. }
    15. }
    16. public class ReflectDemo1 {
    17. public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
    18. // method1();
    19. //method2();
    20. //method3();
    21. //method4();
    22. }
    23. private static void method4() throws ClassNotFoundException, NoSuchFieldException {
    24. // Field getDeclaredField(String name):返回单个成员变量对象
    25. //1.获取class对象
    26. Class clazz = Class.forName("com.itheima.myreflect4.Student");
    27. //2.获取money成员变量
    28. Field field = clazz.getDeclaredField("money");
    29. //3.打印一下
    30. System.out.println(field);
    31. }
    32. private static void method3() throws ClassNotFoundException, NoSuchFieldException {
    33. // Field getField(String name):返回单个公共成员变量对象
    34. //想要获取的成员变量必须是真实存在的
    35. //且必须是public修饰的.
    36. //1.获取class对象
    37. Class clazz = Class.forName("com.itheima.myreflect4.Student");
    38. //2.获取name这个成员变量
    39. //Field field = clazz.getField("name");
    40. //Field field = clazz.getField("name1");
    41. Field field = clazz.getField("money");
    42. //3.打印一下
    43. System.out.println(field);
    44. }
    45. private static void method2() throws ClassNotFoundException {
    46. // Field[] getDeclaredFields():返回所有成员变量对象的数组
    47. //1.获取class对象
    48. Class clazz = Class.forName("com.itheima.myreflect4.Student");
    49. //2.获取所有的Field对象
    50. Field[] fields = clazz.getDeclaredFields();
    51. //3.遍历
    52. for (Field field : fields) {
    53. System.out.println(field);
    54. }
    55. }
    56. private static void method1() throws ClassNotFoundException {
    57. // Field[] getFields():返回所有公共成员变量对象的数组
    58. //1.获取class对象
    59. Class clazz = Class.forName("com.itheima.myreflect4.Student");
    60. //2.获取Field对象.
    61. Field[] fields = clazz.getFields();
    62. //3.遍历
    63. for (Field field : fields) {
    64. System.out.println(field);
    65. }
    66. }
    67. }

2.4.2Field类用于给成员变量赋值的方法

  • 方法介绍
    | 方法名 | 说明 | | —- | —- | | void set(Object obj, Object value) | 赋值 | | Object get(Object obj) | 获取值 |

  • 示例代码

    // Student类同上一个示例,这里就不在重复提供了
    public class ReflectDemo2 {
     public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, IllegalAccessException, InstantiationException {
    //        Object get(Object obj) 返回由该 Field表示的字段在指定对象上的值。
         //method1();
         //method2();
    
     }
    
     private static void method2() throws ClassNotFoundException, NoSuchFieldException, InstantiationException, IllegalAccessException {
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect4.Student");
    
         //2.获取成员变量Field的对象
         Field field = clazz.getDeclaredField("money");
    
         //3.取消一下访问检查
         field.setAccessible(true);
    
         //4.调用get方法来获取值
         //4.1创建一个对象
         Student student = (Student) clazz.newInstance();
         //4.2获取指定对象的money的值
         Object o = field.get(student);
    
         //5.打印一下
         System.out.println(o);
     }
    
     private static void method1() throws ClassNotFoundException, NoSuchFieldException, InstantiationException, IllegalAccessException {
         //        void set(Object obj, Object value):给obj对象的成员变量赋值为value
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect4.Student");
    
         //2.获取name这个Field对象
         Field field = clazz.getField("name");
    
         //3.利用set方法进行赋值.
         //3.1先创建一个Student对象
         Student student = (Student) clazz.newInstance();
         //3.2有了对象才可以给指定对象进行赋值
         field.set(student,"zhangsan");
    
         System.out.println(student);
     }
    }
    

反射获取成员方法并使用

Class类获取成员方法对象的方法

  • 方法分类
    | 方法名 | 说明 | | —- | —- | | Method[] getMethods() | 返回所有公共成员方法对象的数组,包括继承的 | | Method[] getDeclaredMethods() | 返回所有成员方法对象的数组,不包括继承的 | | Method getMethod(String name, Class<?>… parameterTypes) | 返回单个公共成员方法对象 | | Method getDeclaredMethod(String name, Class<?>… parameterTypes) | 返回单个成员方法对象 |

  • 示例代码

    public class Student {
    
     //私有的,无参无返回值
     private void show() {
         System.out.println("私有的show方法,无参无返回值");
     }
    
     //公共的,无参无返回值
     public void function1() {
         System.out.println("function1方法,无参无返回值");
     }
    
     //公共的,有参无返回值
     public void function2(String name) {
         System.out.println("function2方法,有参无返回值,参数为" + name);
     }
    
     //公共的,无参有返回值
     public String function3() {
         System.out.println("function3方法,无参有返回值");
         return "aaa";
     }
    
     //公共的,有参有返回值
     public String function4(String name) {
         System.out.println("function4方法,有参有返回值,参数为" + name);
         return "aaa";
     }
    }
    public class ReflectDemo1 {
     public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
         //method1();
         //method2();
         //method3();
         //method4();
         //method5();
     }
    
     private static void method5() throws ClassNotFoundException, NoSuchMethodException {
         //        Method getDeclaredMethod(String name, Class<?>... parameterTypes):
    //                                返回单个成员方法对象
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect5.Student");
         //2.获取一个成员方法show
         Method method = clazz.getDeclaredMethod("show");
         //3.打印一下
         System.out.println(method);
     }
    
     private static void method4() throws ClassNotFoundException, NoSuchMethodException {
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect5.Student");
         //2.获取一个有形参的方法function2
         Method method = clazz.getMethod("function2", String.class);
         //3.打印一下
         System.out.println(method);
     }
    
     private static void method3() throws ClassNotFoundException, NoSuchMethodException {
         //        Method getMethod(String name, Class<?>... parameterTypes) :
    //                                返回单个公共成员方法对象
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect5.Student");
         //2.获取成员方法function1
         Method method1 = clazz.getMethod("function1");
         //3.打印一下
         System.out.println(method1);
     }
    
     private static void method2() throws ClassNotFoundException {
         //        Method[] getDeclaredMethods():
    //                                返回所有成员方法对象的数组,不包括继承的
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect5.Student");
    
         //2.获取Method对象
         Method[] methods = clazz.getDeclaredMethods();
         //3.遍历一下数组
         for (Method method : methods) {
             System.out.println(method);
         }
     }
    
     private static void method1() throws ClassNotFoundException {
         //        Method[] getMethods():返回所有公共成员方法对象的数组,包括继承的
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect5.Student");
         //2.获取成员方法对象
         Method[] methods = clazz.getMethods();
         //3.遍历
         for (Method method : methods) {
             System.out.println(method);
         }
     }
    }
    

Method类用于执行方法的方法

  • 方法介绍
    参数一: 用obj对象调用该方法
    参数二: 调用方法的传递的参数(如果没有就不写)
    返回值: 方法的返回值(如果没有就不写) | 方法名 | 说明 | | —- | —- | | Object invoke(Object obj, Object… args) | 运行方法 |

  • 示例代码

    public class ReflectDemo2 {
     public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InstantiationException, InvocationTargetException {
    //        Object invoke(Object obj, Object... args):运行方法
    //        参数一:用obj对象调用该方法
    //        参数二:调用方法的传递的参数(如果没有就不写)
    //        返回值:方法的返回值(如果没有就不写)
    
         //1.获取class对象
         Class clazz = Class.forName("com.itheima.myreflect5.Student");
         //2.获取里面的Method对象  function4
         Method method = clazz.getMethod("function4", String.class);
         //3.运行function4方法就可以了
         //3.1创建一个Student对象,当做方法的调用者
         Student student = (Student) clazz.newInstance();
         //3.2运行方法
         Object result = method.invoke(student, "zhangsan");
         //4.打印一下返回值
         System.out.println(result);
     }
    }