反射
-
反射_获取字节码Class对象的三种方式
-
ReflectDemo1
package cn.xiaoge.day22.reflect; /* 反射: 框架设计的灵魂 框架: 半成品软件. 可以在框架的基础上进行软件开发, 简化编码 反射: 将类的各个组成部分封装为其他对象, 这就是反射机制 好处: 1. 可以在程序运行过程中, 操作这些对象 2. 可以解耦, 提高程序的可扩展性 获取class对象的方式: 1. Class.forName("全类名"): (全类名--->包名.类名)将字节码文件加载进内存, 返回Class对象 - 多用于配置文件, 将类名定义在配置文件中. 读取文件, 加载类 2. 类名.class: 通过类名的属性class获取 - 多用于参数的传递 3. 对象.getClass(): getClass()方法在Object类中定义着. - 多用于对象的获取字节码的方式 结论: 同一个字节码文件(*.class)在一次程序运行过程中, 只会被加载一次, 不论通过哪一种方式获取的Class对象都是同一个. Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的 获取功能: 1. 获取成员变量们 - Field[] getFields(): 获取所有public修饰的成员变量的 - Field getField(String name): 获取指定名称的 public修饰的成员变量的 - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符 - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符 2. 获取构造方法们 - Constructor<?>[] getConstructors() - Constructor<T> getConstructor(类<?>... parameterTypes) - Constructor<?>[] getDeclaredConstructors() - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes) 3. 获取成员方法们 - Method[] getMethods() - Method getMethod(String name, 类<?>... parameterTypes) - Method[] getDeclaredMethods() - Method getDeclaredMethod(String name, 类<?>... parameterTypes) 4. 获取类名 - String getName() Field: 成员变量 操作: 1. 设置值 - void set(Object obj, Object value) 2. 获取值 - get(Object obj) 3. 忽略访问权限修饰符的安全检查 - d.setAccessible(true): 暴力反射 Constructor: 构造方法 创建对象: - T newInstance(Object... initargs) - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法 Method: 方法对象 执行方法: - Object invoke(Object obj, Object... args) 获取方法名称: - String getName(): 获取方法名 获取类名: Class对象.getName(); */ import cn.xiaoge.day22.domain.Person; import cn.xiaoge.day22.domain.Student; public class ReflectDemo1 { /** * 获取class对象的方式: * 1. Class.forName("全类名"): (全类名--->包名.类名)将字节码文件加载进内存, 返回Class对象 * 2. 类名.class: 通过类名的属性class获取 * 3. 对象.getClass(): getClass()方法在Object类中定义着. */ public static void main(String[] args) throws Exception { // 1. Class.forName("全类名"): (全类名--->包名.类名)将字节码文件加载进内存, 返回Class对象 Class cls1 = Class.forName("cn.xiaoge.day22.domain.Person"); System.out.println(cls1); // 2. 类名.class: 通过类名的属性class获取 Class cls2 = Person.class; System.out.println(cls2); // 3. 对象.getClass(): getClass()方法在Object类中定义着. Class cls3 = new Person().getClass(); System.out.println(cls3); // == 比较三个对象 System.out.println(cls1 == cls2); // true System.out.println(cls1 == cls3); // true Class c = Student.class; System.out.println(c == cls1); // false } } // 运行结果 class cn.xiaoge.day22.domain.Person class cn.xiaoge.day22.domain.Person class cn.xiaoge.day22.domain.Person true true false
-
-
反射_Class对象功能获取Field
-
ReflectDemo2
package cn.xiaoge.day22.reflect; import cn.xiaoge.day22.domain.Person; import java.lang.reflect.Field; public class ReflectDemo2 { private Class personClass; /* Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的 获取功能: 1. 获取成员变量们 - Field[] getFields(): 获取所有public修饰的成员变量的 - Field getField(String name): 获取指定名称的 public修饰的成员变量的 - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符 - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符 2. 获取构造方法们 - Constructor<?>[] getConstructors() - Constructor<T> getConstructor(类<?>... parameterTypes) - Constructor<?>[] getDeclaredConstructors() - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes) 3. 获取成员方法们 - Method[] getMethods() - Method getMethod(String name, 类<?>... parameterTypes) - Method[] getDeclaredMethods() - Method getDeclaredMethod(String name, 类<?>... parameterTypes) 4. 获取类名 - String getName() Field: 成员变量 操作: 1. 设置值 - void set(Object obj, Object value) 2. 获取值 - get(Object obj) 3. 忽略访问权限修饰符的安全检查 - d.setAccessible(true): 暴力反射 Constructor: 构造方法 创建对象: - T newInstance(Object... initargs) - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法 Method: 方法对象 执行方法: - Object invoke(Object obj, Object... args) 获取方法名称: - String getName(): 获取方法名 获取类名: Class对象.getName(); */ public static void main(String[] args) throws Exception { // 1. 获取Person的Class对象 Class personClass = Person.class; /* 1. 获取成员变量们 - Field[] getFields() : 获取所有public修饰的成员变量的 - Field getField(String name): 获取指定名称的 public修饰的成员变量的 - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符 - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符 Field: 成员变量 操作: 1. 设置值 - void set(Object obj, Object value) 2. 获取值 - get(Object obj) 3. 忽略访问权限修饰符的安全检查 - d.setAccessible(true): 暴力反射 */ // 1. Field[] getFields() : 获取所有public修饰的成员变量的 Field[] fields = personClass.getFields(); for (Field field : fields) { System.out.println(field); } System.out.println("----------------------"); // 2. Field getField(String name): 获取指定名称的 public修饰的成员变量的 Field a = personClass.getField("a"); // 获取成员变量a的值 Person p = new Person(); Object value = a.get(p); System.out.println(value); // 设置成员变量a的值 a.set(p, "10"); System.out.println(p); System.out.println("==================================="); // 3. Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符 Field[] declaredFields = personClass.getDeclaredFields(); for (Field declaredField : declaredFields) { System.out.println(declaredField); } // 4. Field getDeclaredField(String name) Field d = personClass.getDeclaredField("d"); // 忽略访问权限修饰符的安全检查 d.setAccessible(true); // 暴力反射 Object value2 = d.get(p); System.out.println(value2); } } // 运行结果 public java.lang.String cn.xiaoge.day22.domain.Person.a ---------------------- null Person{name='null', age=0, a='10', b='null', c='null', d='null'} =================================== private java.lang.String cn.xiaoge.day22.domain.Person.name private int cn.xiaoge.day22.domain.Person.age public java.lang.String cn.xiaoge.day22.domain.Person.a protected java.lang.String cn.xiaoge.day22.domain.Person.b java.lang.String cn.xiaoge.day22.domain.Person.c private java.lang.String cn.xiaoge.day22.domain.Person.d null
-
-
反射_Class对象功能获取Constructor
-
ReflectDemo3
package cn.xiaoge.day22.reflect; import cn.xiaoge.day22.domain.Person; import java.lang.reflect.Constructor; public class ReflectDemo3 { private Class personClass; /* Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的 获取功能: 1. 获取成员变量们 - Field[] getFields(): 获取所有public修饰的成员变量的 - Field getField(String name): 获取指定名称的 public修饰的成员变量的 - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符 - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符 2. 获取构造方法们 - Constructor<?>[] getConstructors() - Constructor<T> getConstructor(类<?>... parameterTypes) - Constructor<?>[] getDeclaredConstructors() - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes) 3. 获取成员方法们 - Method[] getMethods() - Method getMethod(String name, 类<?>... parameterTypes) - Method[] getDeclaredMethods() - Method getDeclaredMethod(String name, 类<?>... parameterTypes) 4. 获取类名 - String getName() Field: 成员变量 操作: 1. 设置值 - void set(Object obj, Object value) 2. 获取值 - get(Object obj) 3. 忽略访问权限修饰符的安全检查 - d.setAccessible(true): 暴力反射 Constructor: 构造方法 创建对象: - T newInstance(Object... initargs) - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法 Method: 方法对象 执行方法: - Object invoke(Object obj, Object... args) 获取方法名称: - String getName(): 获取方法名 获取类名: Class对象.getName(); */ public static void main(String[] args) throws Exception { // 1. 获取Person的Class对象 Class personClass = Person.class; /* 2. 获取构造方法们 - Constructor<?>[] getConstructors() - Constructor<T> getConstructor(类<?>... parameterTypes) - Constructor<?>[] getDeclaredConstructors() - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes) */ // Constructor<T> getConstructor(类<?>... parameterTypes) Constructor c = personClass.getConstructor(String.class, int.class); System.out.println(c); // 创建对象 Object person = c.newInstance("张三", 23); System.out.println(person); System.out.println("====================================="); // Constructor<T> getConstructor(类<?>... parameterTypes) Constructor c1 = personClass.getConstructor(); System.out.println(c); // 创建对象 Object person1 = c1.newInstance(); System.out.println(person1); System.out.println("====================================="); // 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法 Object person2 = personClass.newInstance(); System.out.println(person2); System.out.println("====================================="); Constructor[] cs = personClass.getConstructors(); for (Constructor constructor : cs) { System.out.println(constructor); } Object person3 = cs[0].newInstance(); System.out.println(person3); Object person4 = cs[1].newInstance("李四", 33); System.out.println(person4); } } // 运行结果 public cn.xiaoge.day22.domain.Person(java.lang.String,int) Person{name='张三', age=23, a='null', b='null', c='null', d='null'} ===================================== public cn.xiaoge.day22.domain.Person(java.lang.String,int) Person{name='null', age=0, a='null', b='null', c='null', d='null'} ===================================== Person{name='null', age=0, a='null', b='null', c='null', d='null'} ===================================== public cn.xiaoge.day22.domain.Person() public cn.xiaoge.day22.domain.Person(java.lang.String,int) Person{name='null', age=0, a='null', b='null', c='null', d='null'} Person{name='李四', age=33, a='null', b='null', c='null', d='null'}
-
-
反射_Class对象功能获取Method
-
ReflectDemo4
package cn.xiaoge.day22.reflect; import cn.xiaoge.day22.domain.Person; import java.lang.reflect.Constructor; import java.lang.reflect.Method; public class ReflectDemo4 { private Class personClass; /* Class对象功能: 不带Declared只能获取公共的就是public的, 带Declared就是获取所有的 获取功能: 1. 获取成员变量们 - Field[] getFields(): 获取所有public修饰的成员变量的 - Field getField(String name): 获取指定名称的 public修饰的成员变量的 - Field[] getDeclaredFields(): 获取所有的成员变量, 不考虑修饰符 - Field getDeclaredField(String name): 获取指定的成员变量, 不考虑修饰符 2. 获取构造方法们 - Constructor<?>[] getConstructors() - Constructor<T> getConstructor(类<?>... parameterTypes) - Constructor<?>[] getDeclaredConstructors() - Constructor<T> getDeclaredConstructor(类<?>... parameterTypes) 3. 获取成员方法们 - Method[] getMethods() - Method getMethod(String name, 类<?>... parameterTypes) - Method[] getDeclaredMethods() - Method getDeclaredMethod(String name, 类<?>... parameterTypes) 4. 获取类名 - String getName() Field: 成员变量 操作: 1. 设置值 - void set(Object obj, Object value) 2. 获取值 - get(Object obj) 3. 忽略访问权限修饰符的安全检查 - d.setAccessible(true): 暴力反射 Constructor: 构造方法 创建对象: - T newInstance(Object... initargs) - 如果使用空参数构造方法创建对象, 操作可以简化: Class对象的newInstance方法 Method: 方法对象 执行方法: - Object invoke(Object obj, Object... args) 获取方法名称: - String getName(): 获取方法名 获取类名: Class对象.getName(); */ public static void main(String[] args) throws Exception { // 1. 获取Person的Class对象 Class personClass = Person.class; /* 3. 获取成员方法们 - Method[] getMethods() - Method getMethod(String name, 类<?>... parameterTypes) - Method[] getDeclaredMethods() - Method getDeclaredMethod(String name, 类<?>... parameterTypes) */ // 获取指定名称的方法 Method eat_method = personClass.getMethod("eat"); Person p = new Person(); // 执行方法 eat_method.invoke(p); System.out.println("========================="); Method eat_method2 = personClass.getMethod("eat", String.class); // 执行方法 eat_method2.invoke(p, "鸡"); System.out.println("========================="); // 获取所有public修饰的方法 Method[] methods = personClass.getMethods(); for (Method method : methods) { System.out.println(method); System.out.println(method.getName()); } System.out.println("========================="); // 获取类名 String className = personClass.getName(); System.out.println(className); } } // 运行结果 eat.... ========================= eat:鸡 ========================= public java.lang.String cn.xiaoge.day22.domain.Person.toString() toString public java.lang.String cn.xiaoge.day22.domain.Person.getName() getName public void cn.xiaoge.day22.domain.Person.setName(java.lang.String) setName public int cn.xiaoge.day22.domain.Person.getAge() getAge public void cn.xiaoge.day22.domain.Person.setAge(int) setAge public void cn.xiaoge.day22.domain.Person.eat(java.lang.String) eat public void cn.xiaoge.day22.domain.Person.eat() eat public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException wait public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException wait public final void java.lang.Object.wait() throws java.lang.InterruptedException wait public boolean java.lang.Object.equals(java.lang.Object) equals public native int java.lang.Object.hashCode() hashCode public final native java.lang.Class java.lang.Object.getClass() getClass public final native void java.lang.Object.notify() notify public final native void java.lang.Object.notifyAll() notifyAll ========================= cn.xiaoge.day22.domain.Person
-
-
案例
-
配置文件pro.properties
className=cn.xiaoge.day22.domain.Student methodName=sleep
-
domain包下的两个类
- Person
package cn.xiaoge.day22.domain; public class Person { private String name; private int age; public String a; protected String b; String c; // 默认default private String d; public Person() { } public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public void eat(){ System.out.println("eat...."); } public void eat(String food){ System.out.println("eat:" + food); } @Override public String toString() { return "Person{" + "name='" + name + '\'' + ", age=" + age + ", a='" + a + '\'' + ", b='" + b + '\'' + ", c='" + c + '\'' + ", d='" + d + '\'' + '}'; } }
- Student
package cn.xiaoge.day22.domain; public class Student { public void sleep(){ System.out.println("sleep..."); } }
- Person
-
ReflectTest
package cn.xiaoge.day22.reflect; /* 需求: 写一个"框架", 不能改变该类的任何代码, 可以帮我们创建任意类的独享, 并且执行其中任意方法 实现: 1. 配置文件 2. 反射 步骤: 1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中 2. 在程序中加载读取配置文件 3. 使用反射技术来加载类文件进内存 4. 创建对象 5. 执行方法 */ import java.io.InputStream; import java.lang.reflect.Method; import java.util.Properties; public class ReflectTest { /** * 框架类 */ public static void main(String[] args) throws Exception { // 可以创建任意类的对象, 可以执行任意方法 /* 前提: 不能改变该类的任何代码, 可以创建任意类的对象, 可以执行任意方法 */ // 1. 加载配置文件, 使用Properties里面的load方法把pro.properties加载到内存中, 返回一个集合 // 1.1 创建Properties对象 Properties pro = new Properties(); // 1.2 加载配置文件, 转换为一个集合 // 1.2.1 获取class目录下的配置文件 ClassLoader classLoader = ReflectTest.class.getClassLoader();// 获取这个字节码文件对应的类加载器 InputStream is = classLoader.getResourceAsStream("pro.properties");// 返回获取资源对应的字节流 pro.load(is); // 2. 获取配置文件中定义的数据 获取对应文件里面的键所对应的值 String className = pro.getProperty("className"); String methodName = pro.getProperty("methodName"); // 3. 加载该类文件进内存 Class cls = Class.forName(className); // 4. 创建对象 Object obj = cls.newInstance(); // 5. 获取方法对象 Method methodObj = cls.getMethod(methodName); // 6. 执行方法 methodObj.invoke(obj); } } // 运行结果 sleep...
-