各自努力,最高处见!加油!
一、引入
根据配置文件re.properties指定信息,创建Cat对象并调用方法hi。
classfullpath=com.hspedu.Cat
method=hi
这样的需求在学习框架的时候特别多,即通过外部文件配置,在不修改源码的情况下,来控制情绪,也符合设计模式的OCP原则(开闭原则:不修改源代码,来扩展功能)
二、反射机制
- 反射机制允许程序在执行期借助于Reflection API获取任何类的内部信息(比如成员变量,构造器,成员方法等),并能操作对象的属性及方法。反射在设计模式和框架底层都会用到。
- 加载完类之后,在堆中就产生了一个Class类型(这个类名就是Class)的对象(一个类只有一个Class对象),这个对象包含了类的完整结构信息。通过这个对象得到类的结构。这个对象就像一面镜子,透过这个镜子看到类的结构,所以,形象称之为:反射。
![在这里插入图片描述](https://img-blog.csdnimg.cn/5ea38bfb5bf6430d83f631d2bdbc82ad.png?x-oss-process=image/watermark,type_ZHJvaWRzYW5zZmFsbGJhY2s,shadow_50,text_Q1NETiBATUjmoqblm54=,size_20,color_FFFFFF,t_70,g_se,x_16
Java反射机制可以完成:
- 在运行时判断任意一个对象所属的类
- 在运行时构造任意一个类的对象
- 在运行时得到任意一个类所具有的成员变量和方法
- 在运行时调用任意一个对象的成员变量和方法
- 生成动态代理
三、反射相关类
反射相关的主要类:
- java.lang.Class:代表一个类,Class对象表示某个类加载后再堆中存在对象
- java.lang.reflect.Method:代表类的方法,Method对象表示某个类的方法
- java.lang.reflect.Field:代表类的相关变量,Field对象表示某个类的成员变量
- java.lang.reflect.Constructor:代表类的构造方法,Constructor对象表示构造器
示例代码:
import java.io.FileInputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Properties;
public class Reflection01 {
public static void main(String[] args) throws Exception{
//1. 使用Properties 类, 可以读写配置文件
Properties properties = new Properties();
properties.load(new FileInputStream("src\\re.properties"));
String classfullpath = properties.get("classfullpath").toString();//"com.xmtedu.Cat"
String methodName = properties.get("method").toString();//"hi"
//使用反射机制
//得到一个类:文件中存有类的具体路径,通过路径加载类,返回Class类型 类对象cls-->装着Cat这个类的对象
Class cls=Class.forName(classfullpath);
//得道一个类的实例(对象):通过cls得到你加载的类com.xmt.Cat的对象实例 -->将Cat这个类建立对象,虽然用Object接收,但是底层运行的是Cat类型
Object o = cls.newInstance();
System.out.println("o的运行类型="+o.getClass());
//得到某个类的方法:通过cls得到你加载的类 com.xmtedu.Cat的methodName"hi" 的方法对象
Method method = cls.getMethod(methodName);
System.out.println("==========================");
//通过method调用方法:通过装有方法的对象来调用方法
method.invoke(o);
//获取成员变量 getField不能得到私有属性
Field nameField = cls.getField("age");
System.out.println(nameField.get(o));//传统写法:对象.成员变量 反射写法: 成员变量对象.get(对象)
//得到构造器:
Constructor constructor = cls.getConstructor();//括号中可以指定构造器参数类型,否则返回无参构造器
System.out.println(constructor);
Constructor constructor1 = cls.getConstructor(String.class);//有形参的构造器
System.out.println(constructor1);
}
}
四、反射调用优化
反射的优点和缺点:
- 优点:可以动态的创建和使用对象(框架的核心),使用灵活。
- 缺点:使用反射基本是解释执行,对执行速度有影响。
测试传统方法调用函数与反射调用函数的时间差
package reflection;
import com.xmtedu.Cat;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class reflection02 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, InstantiationException, IllegalAccessException {
m1();
m2();
m3();
}
public static void m1(){
Cat cat=new Cat();
long start=System.currentTimeMillis();
for (int i = 0; i < 9000000; i++) {
cat.hi();
}
long end = System.currentTimeMillis();
System.out.println("传统方法调用hi方法90000000次用时:"+(end-start));
}
public static void m2() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
Class cls = Class.forName("com.xmtedu.Cat");
Object o = cls.newInstance();
Method method = cls.getMethod("hi");
long start=System.currentTimeMillis();
for (int i = 0; i < 9000000; i++) {
method.invoke(o);
}
long end = System.currentTimeMillis();
System.out.println("反射方法调用hi方法90000000次用时:"+(end-start));
}
public static void m3() throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException {
Class cls = Class.forName("com.xmtedu.Cat");
Object o = cls.newInstance();
Method method = cls.getMethod("hi");
method.setAccessible(true);//在反射调用方法时,取消访问检查
long start=System.currentTimeMillis();
for (int i = 0; i < 9000000; i++) {
method.invoke(o);
}
long end = System.currentTimeMillis();
System.out.println("反射方法调用hi方法(取消访问检查)90000000次用时:"+(end-start));
}
}
反射调用优化——关闭访问检查
- Method和Field、Constructor对象都有setAccessible()方法
- setAccessible作用是启动和禁用访问安全检查的开关
- 参数值为true表示反射的对象在使用时取消访问检查,提高反射的效率。参数值为false则表示反射的对象执行访问检查。