介绍

java反射可以在运行时获取对象的成员和属性, 并且可以动态的创建对象并调用对象的属性.
反射一般编程中很少使用,但是在很多框架中都使用了反射, 比如配置Spring的Xml配置文件中, 就使用全类名配置方式, 其实就是反射的一种使用方式.
同时反射对单例模式有一定的影响, 可以参考反射获取单例对象

获取反射对象

获取反射Class对象一共三种方式

  1. // 1. 使用实例获取
  2. User user = new User();
  3. Class<? extends User> aClass = user.getClass();
  4. // 2. 使用类获取
  5. Class<User> userClass = User.class;
  6. // 3. 全类名获取, 可能会抛出 ClassNotFoundException 异常
  7. Class<?> aClass1 = Class.forName("com.liuzhihang.tool.reflect.User");

获取属性

  1. 获取字段

    1. // 获取所有公有字段 (public)
    2. Field[] fields = aClass.getFields();
    3. // 获取所有字段 (public 缺省, protected, private)
    4. Field[] fields = aClass.getDeclaredFields()
    5. // 获取指定公共字段
    6. Field age = aClass.getField("age");
    7. // 获取指定字段 (public 缺省, protected, private)
    8. Field userName = aClass.getDeclaredField("userName");

    获取构造

  2. 获取构造

    1. // 获取所有构造 不能获取私有
    2. Constructor<?>[] constructors = aClass.getConstructors();
    3. // 获取指定参数类型的构造 不能获取私有 空则获取空参构造 getConstructor(Class<?>... parameterTypes)
    4. Constructor<User> constructor = aClass.getConstructor(String.class);
    5. // 获取所有构造 包含私有
    6. Constructor<?>[] declaredConstructors = aClass.getDeclaredConstructors();
    7. // 获取指定参数类型的构造 可以获取私有 空则获取空参构造 getDeclaredConstructor(Class<?>... parameterTypes)
    8. Constructor<User> declaredConstructor = aClass.getDeclaredConstructor(String.class);
  3. 使用构造创建对象

可以通过 constructor.setAccessible(true); 暴力破解忽略访问修饰符, 来使用私有构造参数

  1. Constructor<User> constructor = aClass.getDeclaredConstructor(String.class);
  2. // 暴力破解
  3. constructor.setAccessible(true);
  4. User test = constructor.newInstance("test");

获取方法

  1. 获取方法

    1. // 获取所有公共方法(包含父类)
    2. Method[] methods = aClass.getMethods();
    3. // 获取所有方法
    4. Method[] methods = aClass.getDeclaredMethods();
    5. // 获取私有方法 第一个参数填方法名称
    6. Method address = aClass.getDeclaredMethod("setAddress", String.class);
    7. // 获取公共方法
    8. Method address = aClass.getMethod("setAddress", String.class);
  2. 使用方法

    1. Class<?> aClass = Class.forName("com.liuzhihang.tool.reflect.User")
    2. Method address = aClass.getDeclaredMethod("setAddress", String.class);
    3. User user = aClass.getConstructor().newInstance();
    4. System.out.println(user.toString());
    5. // 解除私有限制
    6. address.setAccessible(true);
    7. // 使用invoke来调用方法
    8. address.invoke(user, "北京");
    9. System.out.println(user.toString());

    获取其他属性

    还可以获取类实现的接口, 父类, 注解, 以及判断类的类型等多种使用方式.