反射,指在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法,对任意一个对象,都能调用它的任意一个方法。这种动态获取信息,以及动态调用对象方法的功能,叫做java语言的反射机制。反射很强大,有优点也有缺点。
优点:灵活性高。因为反射属于动态编译,即只有到运行时才动态创建 &获取对象实例。
缺点:执行效率低
Class类:代表一个类。
Field类:代表类的成员变量(成员变量也称为类的属性)。
Method类:代表类的方法。
Constructor类:代表类的构造方法。
基础类 |
package cn.yx.zg.反射.demo3; public class Person { private String name; private int age; public Person() { super(); } public Person(String name, int age) { super(); 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; } @Override public String toString() { return "Person [name=" + name + ", age=" + age + "]"; } public void eat() { System.out.println("调用无参方法"); } public void eat(int num) { System.out.println("调用有参方法"); } }
1、获取Class的三种方式 |
public static void main(String[] args) throws Exception { Class<Person> clazz1 = (Class<Person>) Class.forName("cn.yx.zg.反射.demo3.Person"); Class clazz2 = Person.class; Person p = new Person(); Class clazz3 = p.getClass(); System.out.println(clazz1 == clazz2);//true System.out.println(clazz2 == clazz3);//true //常用的方式 try { Class npcClazz3 = Class.forName("com.反射机制.demo1.Person"); } catch (ClassNotFoundException e) { e.printStackTrace(); } }
2、newInstance方法获得对象 |
Class类的newInstance()方法是使用该类无参的构造函数创建对象, 如果一个类没有无参的构造函数, 就不能这样创建了,可以调用Class类的getConstructor (String.class,int.class)方法获取一个指定的构造函数然后再调用
public static void main(String[] args) throws Exception { Class clazz = Class.forName("cn.yx.zg.反射.demo3.Person"); //通过无参构造创建对象 Person p = (Person) clazz.newInstance(); System.out.println(p); //通过构造方法参数创建对象 Constructor c = clazz.getConstructor(String.class,int.class); //获取有参构造 Person p2 = (Person) c.newInstance("张三",23); //通过有参构造创建对象 System.out.println(p2); }
3、Field方法操作字段。动态给字段设置值,获取值 |
Class.getField(String)方法可以获取类中的指定字段(可见的), 如果是私有的可以用getDeclaedField(“name”)方法获取,通过set(obj, “李四”)方法可以设置指定对象上该字段的值, 如果是私有的需要先调用setAccessible(true)设置访问权限,用获取的指定的字段调用get(obj)可以获取指定对象中该字段的值
public static void main(String[] args) throws Exception { Class clazz = Class.forName("cn.yx.zg.反射.demo3.Person"); Constructor c = clazz.getConstructor(String.class, int.class); //获取有参构造 Person p = (Person) c.newInstance("张刚", 2); //通过有参构造创建对象 //Field f = clazz.getField("name"); //获取姓名字段 //f.set(p, "李四"); //修改姓名的值 Field f = clazz.getDeclaredField("name"); //如果name字段是private权限 暴力反射获取字段 f.setAccessible(true); //去除私有权限 f.set(p, "李四"); //name字段的类型 Class<?> type = f.getType(); System.out.println(type); System.out.println(p); }
4、invoke方法 动态调用指定方法 |
Class.getMethod(String, Class…) 和Class.getDeclaredMethod(String, Class…)方法可以获取类中的指定方法,调用invoke(Object, Object…)可以调用该方法
public static void main(String[] args) throws Exception { Class clazz = Class.forName("cn.yx.zg.反射.demo3.Person"); Constructor c = clazz.getConstructor(String.class, int.class); //获取有参构造 Person p = (Person) c.newInstance("张三", 23); //通过有参构造创建对象 Method m = clazz.getMethod("eat"); //获取eat方法 m.invoke(p); Method m2 = clazz.getMethod("eat", int.class); //获取有参的eat方法 m2.invoke(p, 10); }
反射机制是很多Java框架的基石!!!