一.基本概念
反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
二.反射主要能做些什么
-
在运行时判断任意一个对象所属的类;
-
在运行时构造任意一个类的对象;
-
在运行时判断任意一个类所具有的成员变量和方法;
-
在运行时调用任意一个对象的方法;
三.通过对象获取完整类名的三种方式
说明:反射之中的泛型都用<?>表示
1.通过Object类的getClass()方法取得
class Person {
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Person per = new Person(); // 正着操作
Class<?> cls = per.getClass(); // 取得Class对象
System.out.println(cls.getName()); // 反着来
}
}
2.使用“类.class”取得
class Person {
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Person.class; // 取得Class对象
System.out.println(cls.getName());
}
}
3.使用Class类内部定义的一个static方法
class Person {
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person") ; // 使用类的全类名取得Class对象
System.out.println(cls.getName());
}
}
四.通过反射实例化对象
class Person {
@Override
public String toString() {
return "Person Class Instance .";
}
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person"); // 取得Class对象
Object obj = cls.newInstance(); // 实例化对象,和使用关键字new一样
Person per = (Person) obj; // 向下转型
System.out.println(per); // 调用toString()
}
}
五.通过反射机制实现工厂设计模式
对于普通的工厂模式当我们在添加一个子类的时候,就需要对应的修改工厂类。
当我们添加很多的子类的时候,会很麻烦。现在我们利用反射机制实现工厂模式,可以在不修改工厂类的情况下添加任意多个子类。
interface Fruit {
public void eat() ;
}
class Apple implements Fruit {
public void eat() {
System.out.println("吃苹果。");
};
}
class Factory {
public static Fruit getInstance(String className) {
Fruit f = null ;
try {
f = (Fruit) Class.forName(className).newInstance() ; // 反射实例化
} catch (Exception e) {
e.printStackTrace();
}
return f ;
}
}
public class FactoryDemo {
public static void main(String[] args) {
Fruit f = Factory.getInstance("cn.kd.demo.Apple") ;
f.eat() ;
}
}
六.取得一个类之中的全部构造
import java.lang.reflect.Constructor;
class Person {
public Person() {}
public Person(String name) {}
public Person(String name,int age) {}
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person") ; // 取得Class对象
Constructor<?> cons [] = cls.getConstructors() ; // 取得全部构造
for (int x = 0; x < cons.length; x++) {
System.out.println(cons[x]);
}
}
}
七.无参构造通过反射实例化对象的操作
class Person {
public Person() {
}
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person"); // 取得Class对象
Object obj = cls.newInstance(); // 实例化对象,默认调用无参构造,如果类中没定义无参构造,此处会报错
System.out.println(obj);
}
}
八.通过反射调用指定参数的构造实例化对象
import java.lang.reflect.Constructor;
class Person {
private String name;
private int age;
public Person(String name, int age) { // 有两个参数的构造方法
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person") ; // 取得Class对象
// 取得指定参数类型的构造方法,传递两个参数类型的Class对象
Constructor<?> cons = cls.getConstructor(String.class,int.class) ;
Object obj = cons.newInstance("张三", 20); // 为构造方法传递参数
System.out.println(obj);
}
}
九.取得一个类之中所定义的全部方法
import java.lang.reflect.Method;
class Person {
private String name;
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person"); // 取得Class对象
Method met[] = cls.getMethods(); // 取得全部方法
for (int x = 0; x < met.length; x++) {
System.out.println(met[x]);
}
}
}
十.利用反射调用类之中的setName()、getName()方法
import java.lang.reflect.Method;
class Person {
private String name;
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person"); // 取得Class对象
Object obj = cls.newInstance(); // 实例化对象
String attribute = "name"; // 调用属性名称
// 分别取得Person类中的setName()方法和getName()方法的Method对象
Method setMet = cls.getMethod("set" + initcap(attribute), String.class);
Method getMet = cls.getMethod("get" + initcap(attribute));
setMet.invoke(obj, "张三"); // 等价于:Person对象.setName("张三")
System.out.println(getMet.invoke(obj)); // 等价于:Person对象.getName()
}
public static String initcap(String str) { // 首字母大写
return str.substring(0, 1).toUpperCase().concat(str.substring(1));
}
}
十一.取得一个类之中的全部属性
import java.lang.reflect.Field;
class Person {
private String name; // 为了方便,只设置一个name属性
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person"); // 取得Class对象
Field field[] = cls.getDeclaredFields(); // 取得全部属性
for (int x = 0; x < field.length; x++) {
System.out.println(field[x]);
}
}
}
十二.利用反射操作类中的属性
import java.lang.reflect.Field;
class Person {
private String name;
}
public class TestDemo {
public static void main(String[] args) throws Exception {
Class<?> cls = Class.forName("cn.kd.demo.Person"); // 取得Class对象
Object obj = cls.newInstance(); // 对象实例化属性才会分配空间
Field nameField = cls.getDeclaredField("name"); // 找到name属性
nameField.setAccessible(true); // 解除封装
nameField.set(obj, "张三"); // Person对象.name = "张三"
System.out.println(nameField.get(obj)); // Person对象.name
}
}
十三.核心PAI总结
//获得构造函数的方法
Constructor getConstructor(Class[] params)//根据指定参数获得public构造器
Constructor[] getConstructors()//获得public的所有构造器
Constructor getDeclaredConstructor(Class[] params)//根据指定参数获得public和非public的构造器
Constructor[] getDeclaredConstructors()//获得public的所有构造器
//获得类方法的方法
Method getMethod(String name, Class[] params),根据方法名,参数类型获得方法
Method[] getMethods()//获得所有的public方法
Method getDeclaredMethod(String name, Class[] params)//根据方法名和参数类型,获得public和非public的方法
Method[] getDeclaredMethods()//获得所以的public和非public方法
// 获得类中属性的方法
Field getField(String name)//根据变量名得到相应的public变量
Field[] getFields()//获得类中所有public的方法
Field getDeclaredField(String name)//根据方法名获得public和非public变量
Field[] getDeclaredFields()//获得类中所有的public和非public方法