一、简介
需求引入:
配置文件中指定创建一个对象(class = Cat)。
即需要从配置文件中提取类名,并创建对象。
基本流程:
//获取配置文件中类名方法名
Properties properties = new Properties();
properties.load(new FileInputStream("D:\\idea_java_projects\\Java_Learning_Exercise\\src\\Reflection\\.properties"));
String className = properties.getProperty("classPath");
String methodName = properties.getProperty("method");
String fieldPubName = properties.getProperty("publicField");
String fieldPriName = properties.getProperty("privateField");
//1.加载反射类,返回这个类的类型的变量,即返回的对象为指定类的类型
Class aClass = Class.forName(className);
//2.通过第一步得到的类类型的变量的newInstance()方法创建对象实例
Object o = aClass.newInstance();
//3.通过第一步得到的类类型的变量的
// ①getMethod()方法得到对象方法,
// ②getField()方法得到成员变量
// ③getConstructor()方法得到构造器
// 即在反射中,把成员方法、变量、构造器都是类的一个对象
// 且正常调用是为 对象.方法 对象.变量 对象.构造器
// 现在变为 方法对象.invoke(Class类对象创建的实例)
//method表示该类的这个方法的对象
Method method = aClass.getMethod(methodName);
//field表示该类对应字段的对象
Field field = aClass.getField(fieldPubName);
//constructor表示该类构造器的对象
// ()内可以指定构造器参数的类型,没有参数表示无参构造器。
Constructor constructor = aClass.getConstructor();//这是无参构造器
Constructor constructor1 = aClass.getConstructor(String.class,int.class);//这是带两个参数的构造器
//4.通过 该方法所属类的对象和
// 方法对象调用方法
// 成员变量对象使用成员变量
// 构造器构造新的对象
method.invoke(o);//传统方法:对象.方法() 反射机制:方法对象.invoke(对象)
System.out.println(field.get(o));//传统方法:对象.属性(对象.get对象名) 反射机制:属性.get(对象)
二、反射机制
1.反射机制允许程序在执行期间借助于Reflection API取得任何类的内部信息(比如成员变量、构造器、成员方法等),
并能操作对象的属性以及方法。
反射在设计模式和框架底层都会用到。
2.加载完类之后,在堆中就产生了一个Class类型的对象(一个类只有一个Class对象),这个对象包含了类的完整信息。
通过这个对象得到类的结构。
这个对象就像一面镜子
,透过这个镜子可以看到类的结构,所以称之为:反射。
反射机制可以完成:
1.在运行时判断任意一个对象所属的类。
2.在运行时构造任意一个类的对象。
3.在运行时得到任意一个类所具有的成员变量和方法。
4.在运行时调用任意一个对象的成员变量和方法。
5.生成动态代理。
1.首先类会进行编译,通过javac指令将.java文件编译成.class字节码文件。
2.通过类加载器ClassLoader,在堆中创建一个Class类型的对象(一个类只能生成一个Class类型的对象),里面存放这个类的成员变量、构造器、成员方法等。
3.然后在堆中创建Cat对象,并且该对象和堆中Class类性的对象互相关联,Cat对象知道自己属于哪个Class对象。
4.通过反射获得Class类型的对象,就可以利用该对象创建对象,调用方法等。
三、反射相关类
1.java.lang.
Class:代表一个类,Class对象表示某个类加载后在堆中的对象。
2.java.lang.reflect.
Method:代表类的方法。
3.java.lang.reflect.
Field:代表类的成员变量。
4.java.lang.reflect.
Constructor:代表类的构造方法。
四、反射优缺点
1.优点:可以动态的创建和使用对象(框架底层的核心),使用灵活。没有反射机制框架技术就失去了底层支持。
2.缺点:使用反射基本是解释执行,对执行速度有影响。
如下图,分别使用传统方法和反射机制调用Cat.hi()方法100000次,比较时间:
优化
:
1.Method和Field、Constructor对象都有setAccessible()方法
2.setAccessible()方法作用是启动和禁用访问安全检查的开关,访问检查,就是查看成员属性、成员方法的使用是否符合访问权限(public、protected、 default、private)。
3.参数值为true表示 反射的对象在使用时取消访问检查,提高反射的效率。
参数值为false表示 反射的对象在使用时执行访问调查。
五、通过反射获取类的结构信息
第一组:java.lang.Class类
第二组:java.lang.reflect.Field类
注意,如果是一个组合关系,即public static 则返回二者Int类型相加的值,即1+8=9
第三组:java,lang.reflect.Method类
第四组:java.lang.reflect.Constructor类
六、利用反射创建实例、操作属性和方法
1.创建实例
第四点就是使用了private修饰的构造器创建对象,可见反射无视了封装性,留了后门。
2.操作属性
3.操作方法
注意,在反射中,如果方法具有返回值,返回时编译类型都是Object类型。