反射
什么是反射机制:
java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,对与任意一个对象,都能够调用他的任意一个方法和属性,这种动态的获取的信息以及动态调用对象的方法的功能成为java反射机制,
class
一旦class文件被加载到内存,就会为其创建一个Class对象。任何类被使用时都会创建一个Class对象。
Class类是Java反射机制的基础,通过Class类,可以得到一个类的基本信息。
静态编译和动态编译
- 静态编译: 在编译时确定类型,绑定对象
- 动态编译: 运行时确定类型,绑定对象
反射机制的优缺点
- 优点:运行期类型的判断,动态加载类,提高代码的灵活度
- 缺点: 性能的瓶颈,反射相当于一系列解释操作,通知JVM要做的事情,性能比直接的java代码要慢的很多
反射机制的应用场景有哪些?
反射是框架设计灵魂
-
在我们平时的项目开发过程中,基本上很少回直接使用到反射机制,但这不能说明反射机制没有用,实际上有很多设计,开发都与反射机制有关,
例如模块话开发,通常反射去调用对应的字节码;动态代理设计模式也采用的反射机制,还有我们日常使用的 Spring/mybatis 等框架也大量使用到了反射机制。
-
举例:
①我们在使用JDBC连接数据库时使用Class.forName()通过反射加载数据库的驱动程序;
②Spring框架也用到很多的反射机制,最经典的就是XML的配置模式,Spring 通过 XML 配置模式装载 Bean 的过程:
- 将程序内所有的 XML 或 Properties 配置文件加载入内存中;
- java类里面解析XML或properties里面的类容,得到对应的实体类的字节码字符串以及相关的属性信息
- 使用反射机制,根据这个字符串获得某个类的Class实例
- 动态配置实例的属性
java获取反射的三种方法
- 通过new对象实现反射机制
- 通过路径实现反射机制
- 通过类名实现反射机制
class类的对象理解
每一个被内加载器到内存中,都会创建一个class的对象,来储存类的信息
反射的起点的就是会的类的class对象
public class Student {
private int id;
String name;
protected boolean sex;
public float score;
}
public class Get {
//获取反射机制三种方式
public static void main(String[] args) throws ClassNotFoundException {
//方式一(通过建立对象)
Student stu = new Student();
Class classobj1 = stu.getClass();
System.out.println(classobj1.getName());
//方式二(所在通过路径-相对路径)
Class classobj2 = Class.forName("fanshe.Student");
System.out.println(classobj2.getName());
//方式三(通过类名)
Class classobj3 = Student.class;
System.out.println(classobj3.getName());
}
}
获得构造法
uclass.getConstructor()
Constructor con = uclass.getConstructor();//获得的是默认无参的构造方法(公共)
User user = (User)con.newInstance();
System.out.println(user);
获得的是默认有参的构造方法(公共)
uclass.getConstructor
Constructor con = uclass.getConstructor(int.class,String.class);//获得的是默认无参的构造方法(公共)
User user = (User)con.newInstance(23,"张三");
System.out.println(user);
获得的是默认无参的构造方法(私有也可以)
uclass.getDeclaredConstructor
con.setAccessible(true);
//设置访问权限
Constructor con = uclass.getDeclaredConstructor(String.class,int.class);//获得的是默认无参的构造方法(私有也可以)
con.setAccessible(true);//设置访问权限
User user = (User)con.newInstance("张三",23);
System.out.println(user)
多个
Constructor[] declaredConstructors = uclass.getDeclaredConstructors();
for (Constructor con : declaredConstructors){
con.setAccessible(true);
}
Field类将类的属性进行封装
可以获得属性的基本信息、属性的值,也可以对属性进行赋值
获得指定的公共的成员变量
uclass.getField("num");
//获得指定的公共的成员变量
uclass.getDeclaredField("name");
获得指定私有的成员变量
Field[] fs = uclass.getDeclaredFields();
for (Field f : fs){
f.setAccessible(true);
System.out.println(f.getName());
}
Method类的作用
Method类将类中的方法进行封装,可以动态获得方法的信息,例如
- getName:获得方法名字
- getParameterTypes:获得方法参数类型
除了动态获取方法信息外,method还能动态调某一个对象的具体放方法
- invoke(Object obj, Object… args) :使用obj调用该方法,参数为args