在Java语言中,除了通过new
关键字来创建类对象的实例,还可以通过反射技术来创建类对象的实例。
通过反射来创建类对象的实例,首先我们得拿到类对象的Class
,如何获取到Class
,请参考以下内容:
在拿到类对象的Class
后,就可以通过Java的反射机制来创建类对象的实例对象了,主要分为以下几种方式:
- Class.newInstance()
- 调用类对象的构造方法
测试类
首先,准备测试类如下:
/**
* @author likly
* @version 1.0
*/
public class Person {
private String name;
private int age;
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;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
Class.newInstance()
测试代码如下:
public class ClassNewInstance {
public static void main(String[] args) throws IllegalAccessException, InstantiationException {
Person person = Person.class.newInstance();
person.setAge(18);
person.setName("likly");
System.out.println(person);
}
}
运行结果为:
Person{name='likly', age=18}
可见,通过Class.newInstance()
方法是可以创建类对象的实例对象的,但是一定要注意,如果把Person
的空参的构造函数去掉,并创建非空的构造函数,再次运行上面的代码,将会抛出如下的异常:
Caused by: java.lang.NoSuchMethodException: likly.java.reflect.Person.<init>()
我们再次添加上空参的构造函数,并修改为如下:
public Person() {
System.out.println("Person");
}
然后再次执行测试程序,运行结果为:
Person
Person{name='likly', age=18}
由此可认识,通过Class.newInstance()
方式创建类对象的对象实例,本质是执行了类对象的默认的空参的构造函数,如果类对象含有非空的构造函数,并且没有显式的声明空参的构造函数,通过Class.newInstance()
方式来创建类对象的实例时,会抛出java.lang.NoSuchMethodException
异常。因此,开发者在设计通过反射创建类对象的对象实例时,一定要判断区分空参的构造方法。
既然Class.newInstance()
方法是通过调用类对象的空参的构造方法来创建类对象实例的,那是不是也可以调用非空的构造方法来创建类对象实例呢?当然是可以的,下面,我们先来说明如何获取类对象的构造方法。
获取类对象的构造方法——Constructor
Constructor
是Java反射机制中的构造函数对象,获取该对象的方法有以下几种:
- Class.getConstructors():获取类对象的所有构造函数
- Class.getConstructor(Class... paramTypes):获取指定的构造函数
获取类对象所有的构造方法并遍历
编写如下的测试代码:
public class ConstructorInstance {
public static void main(String[] args) {
Class p = Person.class;
for(Constructor constructor : p.getConstructors()){
System.out.println(constructor);
}
}
}
运行后结果为:
public likly.java.reflect.Person()
public likly.java.reflect.Person(java.lang.String,int)
获取指定的构造方法
通过Class.getConstructor(Class... paramTypes)
即可获取类对象指定的构造方法,其中paramTypes
为参数类型的Class
可变参数,当不传paramTypes
时,获取的构造方法即为默认的构造方法。
- 获取默认的构造方法
测试代码如下:
public class ConstructorInstance {
public static void main(String[] args) throws Exception {
Class p = Person.class;
Constructor constructor = p.getConstructor();
System.out.println(constructor);
}
}
运行后结果为:
public likly.java.reflect.Person()
- 获取指定参数的构造方法
这里以Person(String,int)
为例
测试代码如下:
public class ConstructorInstance {
public static void main(String[] args) throws Exception {
Class p = Person.class;
Constructor constructor = p.getConstructor(String.class,int.class);
System.out.println(constructor);
}
}
运行后结果为:
public likly.java.reflect.Person(java.lang.String,int)
从以上测试结果可知,可以通过Class.p.getConstructor(Class... paramTypes)
来获取类对象特定的构造方法,如Person(String,int)
。
获取到构造方法有什么用呢?用处当然是来创建类的实例对象了。
通过构造方法创建实例对象
Constructor
对象中有一个方法newInstance(Object ... initargs)
,这里的initargs
即为要传给构造方法的参数,如Person(String,int)
,通过其对应的Constructor
实例,调用newInstance
方法并传入相应的参数,即可通过Person(String,int)
来创建类对象的实例对象。
测试代码如下:
public class ConstructorInstance {
public static void main(String[] args) throws Exception {
Class p = Person.class;
Constructor constructor = p.getConstructor(String.class,int.class);
Person person = (Person) constructor.newInstance("Likly",23);
System.out.println(person); }
}
运行后结果为:
Person{name='Likly', age=23}
到这,创建对象实例的方法又掌握了一种,除了通过new
来创建对象实例,还可以通过反射获取构造方法实例,然后再创建对象实例,不过,这里要注意的是,在使用Class.newInstance()
时一定要注意类对象没有默认的空参的构造函数的情况。
网友评论