Reflection
耦合:模块之间产生了强烈的依赖性的时候,就产生了耦合
解耦:降低模块之间的依赖性(为了模块化开发,便于后续维护)
通过配置文件来动态配置和加载类。
就是把类的字节码看做对象
定义
简单来说,就是动态加载对象,并对对象进行剖析。在java中的反射机制是指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法
对于任意一个对象,都能调用它的任意一个方法,这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制
动态性质
运行时生产对象实例
运行时调用方法
运行时更改属性
API
Class — 代表字节码的类-代表类的类
Package — 代表包的类
Field — 代表属性的类
Method — 代表方法的类
Constructon — 代表构造方法的类
Annotation — 代表注解的类
创建class对象的三种方式
规定不允许我们自己new一个Class对象所以Class对象有这三种创建方式:
①Class name = 类名.class;
如图clz代表的是String类的字节码
②类型 name = 值;
Clas classname= name.getClass();
这里getClass获取了i所对应的类的字节码
③Class name = Classforname(String className);
案例:获取List接口的字节码对象,若参数只写"List"则报错
因为Java有同名类,需要把包名全部写上第三种方法要求传入类的全路径名
如果要用类对象创建实例对象,必须提供一个无参构造
Class<String> aaa = String.class;
String s = aaa.newInstance();
比如Integer类没有无参构造就会报
java.lang.InstantiationException: java.lang.Integer
获取方法
Method name = Class对象.getMethod(String name, Class<?>... parameterTypes)
非公有的构造方法在类为无法执行
因为权限修饰符限制的
非要执行怎么办
打破封锁
暴力破解–暴力拆除
void setAccessible(boolean flag)
获取属性 Field
Field getDeclaredField(String name)
获取实现的所有的接口
Class<?>[] getInterfaces()
获取类名:获取类的全路径名
System.out.println(clz.getName()); //类的全路径名
System.out.println(clz.getCanonicalName()); //类的全路径名
System.out.println(clz.getTypeName()); //类的全路径名
System.out.println(clz.getSimpleName()); //获取类的简称
System.out.println(clz.getSuperclass()); //获取父类的全路径名
获取方法所抛出的异常
Class<?>[] getExceptionTypes()
学习中使用到的反射
了解ArrayList内部扩容时机
了解mybatis装载映射
学习spring中bean的创建
学习jdk动态代理的底层原理
学习SpringMVC拦截器的工作原理
工作中使用到的反射
把相同的业务逻辑进行抽象,形成公用的方法。
反射在ssm中的应用:
- mybatis在进行resultMap或resultType映射时;
- Spring配置文件中bean的创建
- SpringMVC拦截器
- jdk动态代理