一、利用反射创建对象
创建对象:
1、使用 Class 对象的 newInstance()方法创建该 Class 对象的实例,此时该 Class 对象必须要有无参数的构造方法。
2、使用 Class 对象获取指定的 Constructor 对象,再调用 Constructor 的 newInstance()方法创建对象类的实例,此时可以选择使用某个构造方法。如果这个构造方法被私有化起来,那么必须先申请访问,将可以访问设置为 true;
示例代码(最简单的)
class User {
/*private User(){//将默认的构造方法私有化的话就不可以再创建对象,两种方法都是这样
}*/
public String toString() {
return "User对象创建成功!";
}
}
public class ReflectDemo {
public static void main(String[] args) throws Exception {
// 传统方式创建对象
System.out.println(new User());
// 使用反射的方式
Class<User> c = User.class;
User u = c.newInstance();// (直接newInstance的话必须保证默认的构造方法正常存在,也就是没有被私有化!这是前提条件)
System.out.println(u);
}
}
结果截图如下:
复杂点的,更强大的第二种:使用指定构造方法来创建对象:获取该类的 Class 对象。
利用 Class 对象的 getConstructor()方法来获取指定的构造方法。
调用 Constructor 的 newInstance()方法创建对象。
AccessibleObject 对象的 setAccessible(boolean flag)方法,当 flag 为 true 的时候,就会忽略访问
权限(可访问私有的成员)。
其子类有 Field, Method, Constructor;
若要访问对象 private 的成员?在调用之前使用 setAccessible(true),参考http://www.wityx.com/post/1202_1_1.html
Xxx x = getDeclaredXxxx();//才能得到私有的类字段.
总结步骤:
- 获取该类的 Class 对象。
- 利用 Class 对象的 getConstructor()方法来获取指定的构造方法。
- 申请访问(设置为可访问)
- 调用 Constructor(构造方法)的 newInstance()方法创建对象。
示例代码:
import java.lang.reflect.Constructor;
//http://www.wityx.com/post/256_1_1.html
class Per {
private String name;
private int age;
private Per() {
}
private Per(String name) {
}
public String toString() {
return "对象!!!";
}
}
public class ReflectDemo1 {
public static void main(String[] args) throws Exception {
Class<Per> c = Per.class;
// System.out.println(c.newInstance());;//证明利用无参的可以
// 先获得需要被调用的构造器(private 修饰的构造方法)
Constructor<Per> con = c.getDeclaredConstructor();// 调用默认的,什么都不要写
System.out.println(con);
/**
* 现在只需要执行这个构造器,T newInstance(Object... initargs) 使用此 Constructor
* 对象表示的构造方法来创建该构造方法的声明 类的新实例,并用指定的初始化参数初始化该实例。
*/
// 私有的成员是受保护的,不能直接访问(无法直接访问的构造器不一定是私有的),若要访问私有的成员,得先申请一下
//http://www.wityx.com/post/935_1_1.html
con.setAccessible(true);// 允许访问
Per p = con.newInstance();// 成功,通过私有的受保护的构造方法创建了对象
System.out.println("无参构造方法" + p);
con = c.getDeclaredConstructor(String.class);//获取指定的构造方法
System.out.println(con);
con.setAccessible(true);// 允许访问
p = con.newInstance("duixiang");// 成功,通过私有的受保护的构造方法创建了对象
System.out.println("String构造方法" + p);
}
}
备注:对于此时的话,单例模式就不再安全了!反射可破之!!
结果截图:
总结:
(1)getDeclaredConstructor()当没有参数时,它就是默认获取无参构造器,也可以获取指定的构造方法,如con = c.getDeclaredConstructor(String.class);//获取指定的构造方法
(2)T newInstance(Object… initargs) 使用此 Constructor 对象表示的构造方法来创建该构造方法的声明类的新实例,并用指定的初始化参数初始化该实例。
(3)私有的成员是受保护的,不能直接访问(无法直接访问的构造器不一定是私有的),若要访问私有的成员,得先申请一下,如con.setAccessible(true);// 允许访问
二、使用反射调用方法
每个 Method 的对象对应一个具体的底层方法。获得 Method 对象后,程序可以使用 Method 里面的 invoke 方法来执行该底层方法。
Object invoke(Object obj,Object … args):obj 表示调用底层方法的对象,后面的 args 表示传递
的实际参数。
如果底层方法是静态的,那么可以忽略指定的 obj 参数。该参数可以为 null,想想为什么?
如果底层方法所需的形参个数为 0,则所提供的 args 数组长度可以为 0 或 null。
不写,null,或 new Object[]{}
若底层方法返回的是数组类型,invoke 方法返回的不是底层方法的值,而是底层方法的返回类
型;
示例代码:
import java.lang.reflect.Method;
//http://www.wityx.com/post/640_1_1.html
class Dept {
public String show(String name) {// 用反射的方法来调用正常的方法
return name + ",您好!";
}
private void privateshow() {// 用反射来实现对私有化方法的调用
System.out.println("私有的方法:privateshow");
}
public static void staticshow() {
System.out.println("静态的方法:staticshow");
}
}
public class InvokeDemo {
public static void main(String[] args) throws Exception {
//传统方式:
String name = new Dept().show("刘昭");
System.out.println("传统方式:" + name);
// 想要通过反射来调用Dept中的方法
Class<Dept> c = Dept.class;
Method m = c.getMethod("show", String.class);
Object o = m.invoke(c.newInstance(), "刘昭");
System.out.println("反射调用的方式:" + o);
// 私有化的方法
// 参考http://www.wityx.com/post/1205_1_1.html
m = c.getDeclaredMethod("privateshow");// 无参方法
m.setAccessible(true);
o = m.invoke(c.newInstance());
// 静态方法的调用
m = c.getMethod("staticshow");
m.invoke(null);// staticshow为静态方法,不需创建对象,所以这里会是 null
}
}
结果截图:
注:Method getMethod(String name, Class<?>… parameterTypes)返回一个 Method 对象,它反映此 Class 对象所表示的类或接口的指定公共成员方法。name - 方法名;parameterTypes - 参数列表。