反射:框架设计的灵魂
* 框架:半成品软件。可以在框架的基础上进行软件开发,简化编码
* 反射:将类的各个组成部分封装为其他对象,这就是反射机制* 好处:1. 可以在程序运行过程中,操作这些对象。2. 可以解耦,提高程序的可扩展性。
Class类
* 获取Class类的方式:1. Class.forName("全类名"):将字节码文件加载进内存,返回Class类* 多用于配置文件,将类名定义在配置文件中。读取文件,加载类2. 类名.class:通过类名的属性class获取* 多用于参数的传递3. 对象.getClass():getClass()方法在Object类中定义着。* 多用于对象的获取字节码的方式* 结论:同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。
public class People {public void sleep(){System.out.println("睡觉");}
}
public class User {private String name;private int age;public String address;public User() {}public User(String name, int age, String address) {this.name = name;this.age = age;this.address = address;}public String getName() {System.out.println("haha");return name;}public String getName(String value) {System.out.println("haha"+value);return name;}public void setName(String name) {this.name = name;}public void eat(){System.out.println("吃饭饭");}
}public class ReflectClass {public static void main(String[] args) throws ClassNotFoundException {Class<?> class1 &#61; Class.forName("reflect.entity.User");Class<User> class2 &#61; User.class;User user &#61; new User();Class<? extends User> class3 &#61; user.getClass();User user1 &#61; new User();Class<? extends User> class4 &#61; user1.getClass();System.out.println(class1);System.out.println(class2);System.out.println(class3);System.out.println(class4);System.out.println(class1&#61;&#61;class2);System.out.println(class1&#61;&#61;class3);System.out.println(class3&#61;&#61;class4);}
}
* Class对象功能&#xff1a;* 获取功能&#xff1a;1. 获取成员变量们* Field[] getFields() &#xff1a;获取所有public修饰的成员变量* Field getField(String name) 获取指定名称的 public修饰的成员变量* Field[] getDeclaredFields() 获取所有的成员变量&#xff0c;不考虑修饰符* Field getDeclaredField(String name) 2. 获取构造方法们* Constructor>[] getConstructors() * Constructor getConstructor(类>... parameterTypes) * Constructor getDeclaredConstructor(类>... parameterTypes) * Constructor>[] getDeclaredConstructors() 3. 获取成员方法们&#xff1a;* Method[] getMethods() * Method getMethod(String name, 类>... parameterTypes) * Method[] getDeclaredMethods() * Method getDeclaredMethod(String name, 类>... parameterTypes) 4. 获取全类名 * String getName()
Field类
* Field&#xff1a;成员变量* 操作&#xff1a;1. 设置值* void set(Object obj, Object value) 2. 获取值* get(Object obj) 3. 忽略访问权限修饰符的安全检查* setAccessible(true):暴力反射
public class ReflrctField {public static void main(String[] args) throws Exception {Class<User> userClass &#61; User.class;Field[] fields &#61; userClass.getFields();for (Field field : fields) {System.out.println(field);}Field address &#61; userClass.getField("address");User user &#61; new User();Object value &#61; address.get(user);System.out.println(value);address.set(user,"广州");System.out.println(user);System.out.println("&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;");Field[] declaredFields &#61; userClass.getDeclaredFields();for (Field declaredField : declaredFields) {System.out.println(declaredField);}Field age &#61; userClass.getDeclaredField("age");age.setAccessible(true);System.out.println(age.get(user));}
}
Constructor类
* Constructor:构造方法* 创建对象&#xff1a;* T newInstance(Object... initargs) * 如果使用空参数构造方法创建对象&#xff0c;操作可以简化&#xff1a;Class对象的newInstance方法
public class ReflecConstructor {public static void main(String[] args) throws Exception {Class userClass &#61; User.class;Constructor constructor &#61; userClass.getConstructor();System.out.println(constructor);Object user &#61; constructor.newInstance();System.out.println(user);Constructor constructor1 &#61; userClass.getConstructor(String.class, int.class,String.class);System.out.println(constructor1);Object user1 &#61; constructor1.newInstance("hua", 19,"gua");System.out.println(user1);}
}
Method类
* Method&#xff1a;方法对象* 执行方法&#xff1a;* Object invoke(Object obj, Object... args) * 获取方法名称&#xff1a;* String getName:获取方法名
public class ReflectMethod {public static void main(String[] args) throws Exception {Class<User> userClass &#61; User.class;Method nameMethod1 &#61; userClass.getMethod("getName");User user &#61; new User();nameMethod1.invoke(user);Method nameMethod2 &#61; userClass.getMethod("getName", String.class);nameMethod2.invoke(user,"huahuahua");Method[] methods &#61; userClass.getMethods();for (Method method : methods) {System.out.println(method);}System.out.println("&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;");Method[] methods2 &#61; userClass.getDeclaredMethods();for (Method method : methods2) {System.out.println(method);}String userClassName &#61; userClass.getName();System.out.println(userClassName);}
}
案例
* 案例&#xff1a;* 需求&#xff1a;写一个"框架"&#xff0c;不能改变该类的任何代码的前提下&#xff0c;可以帮我们创建任意类的对象&#xff0c;并且执行其中任意方法* 实现&#xff1a;1. 配置文件2. 反射* 步骤&#xff1a;1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中2. 在程序中加载读取配置文件3. 使用反射技术来加载类文件进内存4. 创建对象5. 执行方法在src目录下新建文件 pro.properties,内容如下
className&#61;reflect.entity.People
methodName&#61;sleep
public class ReflectTest {public static void main(String[] args) throws Exception {Properties properties &#61; new Properties();ClassLoader classLoader &#61; ReflectTest.class.getClassLoader();InputStream is &#61; classLoader.getResourceAsStream("user.properties");properties.load(is);String className &#61; properties.getProperty("className");String methodName &#61; properties.getProperty("methodName");Class<?> cls &#61; Class.forName(className);Object obj &#61; cls.newInstance();Method method &#61; cls.getMethod(methodName);method.invoke(obj);}
}