引出反射:

  1. 根据properties配置文件指定信息,创建Cat对象并调用方法hi

classfullpath = com.hspedu.Cat
method=hi
使用现有技术,可以做到吗?

  1. 这样的需求在学习框架时特别多,即通过外部文件配置,在不修改源码情况下,来控制程序,也符合设计模式的ocp原则(开闭原则不修改源码,扩容功能)

classfullpath和 method 可通过properties文件获取
classfullpath = “com.hspedu.Cat”
method = “hi”
image.png

反射机制

Java Reflection

  1. 反射机制允许程序在执行期借助于Reflection API取得任何类的内部信息(比如成员变量,构造器,成员方法等),并能操作对象的属性及方法。反射在设计模式和框架底层都会用到
  2. 加载完类之后,在堆中就产生了一个Class类型的对象(一个类只有一个Class的对象),这个对象包含了类的完整结构信息。通过这个对象得到类的结构。这个Class对象就像一面镜子,透过这个镜子看到类的结构,所以,形象的称之为:反射

Person对象p —> 类型 Person类
Class对象cls —> 类型 Class类
image.png

Java反射机制可以完成

  1. 在运行时判断任意一个对象所属的类
  2. 在运行时构造任意一个类的对象
  3. 在运行时得到任意一个类所具有的成员变量和方法
  4. 在运行时调用任意一个对象的成员变量和方法
  5. 生成动态代理

    反射相关的主要类

  6. java.lang.Class:代表一个类,Class对象表示某个类加载后在堆中的对象

  7. java.lang.reflect.Method:代表类的方法,Method对象表示某个类的方法
  8. java.lang.reflect.Field:代表类的成员变量,Field对象表示某个类的成员变量
  9. java.lang.reflect.Constructor:代表类的构造方法,Constructor对象表示构造器

这些类在 java.lang.reflection

反射的优点和缺点

  1. 优忘:可以动态的创建和使用对象(也是框架底层核心),使用灵活,没有反射机制,框架技术就失去底层支撑
  2. 缺点:使用反射基本是解释执行,对执行速度有影响

    反射调用优化-关闭访问检查

  3. Method和Fied、Constructor对象都有setAccessible()方法

  4. setAccessible()作用是启动和禁用访问安全检查的开关
  5. 参数值为true表示反射的对象在使用时取消访问检查,提高反射的效率。参数值为 false则表示反射的对象执行访问检查

image.png

通过反射获取类的结构信息

java.lang.Class类

  1. getName:获取全类名
  2. getSimpleName:获取简单类名
  3. getFields:获取所有public修饰的属性,包含本类以及父类的
  4. getDeclaredFields:获取本类中所有属性
  5. getMethods:获取所有public修饰的方法,包含本类以及父类的
  6. getDeclaredMethods:获取本类中所有方法
  7. getConstructors:获取本类所有public修饰的构造器
  8. getDeclaredConstructors:取本类中所有构造器
  9. getPackage:以Package形式返回包信息
  10. getSuperClass:以Class式返回父类信息
  11. getInterfaces:以Class[] 形式返回接口信息
  12. getAnnotations:以Annotation[] 形式返回注解信息

    java.lang.reflect.Field类

  13. getModeifiers:以int 形式返回修饰符

[说明:默认修饰符是0,public是1,private是2,protected是4,static是8,
final是16,举例:public(1)+static(8) = 9]

  1. getType:以Class 形式返回类型
  2. getName:返回属性名

    java.lang.reflect.Method类

  3. getModeifiers:以int 形式返回修饰符

[说明:默认修饰符是0,public是1,private是2,protected是4,static是8,
final是16]

  1. getReturnType:以Class 形式获取返回类型
  2. getName:返回方法名
  3. getParameterTypes:以Class[] 返回参数数据类型

    java.lang.reflect.Constructor类

  4. getModifiers:以int形式返回修饰符

  5. getName:返回构造器名(全类名)
  6. getParameterTypes:以Class[] 返回参数类型数组

    通过反射创建对象

  7. 方式1:调用类中的public修饰的无参构造器

  8. 方式2:调用类中的指定构造器
  9. Class类相关方法:
    1. newInstance():调用类中的 无参构造器
    2. getConstructor(Class…clazz):根据参数列表,获取对应的public构造器对象
    3. getDeclaredConstructor(Class…clazz):根据参数列表,获取对应的所有的构造器对象
  10. Constructor类相关方法:
    1. setAccessible:爆破
    2. newInstance(Object…obj):调用构造器

先爆破才能操作私有成员
image.png

通过反射访问类中的成员

访问属性

  1. 根据属性名获取Field对象

Field = Class对象.getDeclredField(属性名);

  1. 爆破:Field对象.setAccessible(true);
  2. 访问:
    1. Field对象.set(o, 值); //o为newInstance出来的对象
    2. syso(Field对象.get(o));
  3. 如果是静态属性,则set和get中的参数o,可以写成null

    访问方法

  4. 根据方法名和参数列表获取Method对象:

Method m = Class对象.getDeclaredMethod(方法名, 形参列表.class);

  1. 获取对象:Object o = Class对象.newInstance();
  2. 爆破:m.setAccessible(true); //私有则爆破
  3. 访问:Object returnValue = m.invoke(o, 实参列表);
  4. 如果是静态方法:则invoke中的参数o,可以写成null