反射
1.为什么要学习反射
静态编译:在编译时确定类型,绑定对象即通过。
动态编译:运行时确定类型,绑定对象。动态编译最大限度地发挥了Java的灵活性,体现了多态的应用,可以减低类之间的耦合性。
java是一门静态语言,反射使java成为准动态语言,反射可以通过一个对象,获取到它的真实类型以及获取该对象里的内容
2.反射
在程序运行过程中,通过字节码对象,去获取到类中的成员信息(构造器,方法,字段),这就称为反射.
2.1 字节码对象
在java中,万物皆对象.每个字节码都有很多共性,每份字节码被jvm加载时会创建字节码的对象存在内存中,每份字节码只有一个对象.当我们拿到了字节码对象,就可以直接操作当前字节码中的构造器,方法,字段.
2.2 获取字节码对象
获取字节码对象的方式:
1.通过Class类的forName(String className)方法获取字节码对象
- className为类的全限定名
- 全限定名:包名.类型
2.通过对象的getClass()方法来获取字节码对象 - new User().getClass()
3.通过类型的class字段来获取字节码对象 - User.class
4.基本数据类型可以通过TYPE属性获取字节码对象 - int.TYPE
第一种方式用得较多
实例:
//通过第一种方式
Class clz = Class.forName("cn.xxxxxx.reflect.Student");
//通过第二种方式
Student stu = new Student();
Class clz = stu1.getClass();
//通过第三种方式
Class clz2 = Student.class;
2.3 获取构造器
使用反射获取构造器:
获取所有得pulic构造器:
Class clz = Person.class;
Constructor[] cons = clz.getConstructors();
获取所有构造器,包括private:
Class clz = Person.class;
Constructor[] cons = clz.getDeclaredConstructors;
获取无参构造器:
Class clz = Person.class;
Constructor con = clz.getConstructor();
获取带参构造器:
Class clz = Person.class;
Constructor con = clz.getConstructor(Long.class,String.class);
获取指定private构造器:
Class clz = Person.class;
Constructor con = clz.getDeclaredConstructor(String.class);
创建对象的快捷方式:
Class clz = Person.class;
Object obj = clz.newInstance();
常见错误:NoSuchMethodException
参数不匹配,找不到指定构造器
2.4 获取方法
使用反射获取方法
获取所有public方法,包括父类的:
Class clz = Person.class;
Method[] methods = clz.getMethods();
获取所有方法,包括private不包括父类的:
Class clz = Person.class;
Method[] methods = clz.getDeclaredMethods();
获取指定参数的public的方法,包括父类的:
Class clz = Person.class;
Method method = clz.getMethod("方法名",String.class);
获取指定参数的private方法:
Class clz = Person.class;
Method method = clz.getDeclaredMethod("方法名",String.class);
调用方法
调用public的方法
Class clz = Person.class;
Object obj = clz.newInstance();
Method method = clz.getMethod("方法名",String.class);
method.invoke(obj,"方法的实参");
调用private的方法
Class clz = Person.class;
Object obj = clz.newInstance();
Method method = clz.getDeclaredMethod("方法名",String.class);
method.setAccessible(true);
method.invoke(obj,"方法的实参");
如果调用的是静态方法,是不需要对象的,所以在invoke方法的第一个参数传递一个null即可.
2.5 操作字段
获取字段
Class clz = Person.class;
Object obj = clz.newInstance();
//因为字段一般都用private修饰,所以用带Declared的方法
//获取全部字段
Feild[] fs = clz.getDeclaredFields();
//获取单个字段
Feild f = clz.gerDeclaredField();
操作字段
//设置字段
f.serAccessible(true);
f.set(obj,"参数");
//获取字段
Object value = f.get(obj);