转载:java中创建对象的方法
Java有4种显式地创建对象的方式:
1. 用new语句创建对象,这是最常用的创建对象的方式。
2. 运用反射手段,调用Java.lang.Class或者java.lang.reflect.Constructor类的newInstance()实例方法。
和Class类的newInstance方法很像, java.lang.reflect.Constructor类里也有一个newInstance方法可以创建对象。我们可以通过这个newInstance方法调用有参数的和私有的构造函数。
3. 调用对象的clone()方法。
要使用clone方法,我们需要先实现Cloneable接口并实现其定义的clone方法。
4. 运用反序列化手段,调用java.io.ObjectInputStream对象的readObject()方法.
下面演示了4种方式创建对象的过程。
public static void main(String[] args) {
Class<?> c=null;
Student Std1=null;
Student Std2=null;
Student Std3=null;
Student Std4=null;
Student Std5=null;
try {
//第一种方式new
Std1 = new Student("Std1", 23);
//第二种方式Java.lang.Class调用无参构造函数
c=Class.forName("zenhobby.Student");
Std2=(Student)c.newInstance();
//第二种方式java.lang.reflect.Constructor调用带参构造函数
Constructor<?>[] cons=c.getConstructors();
Std3=(Student)cons[1].newInstance("Std3",23);
//第三种方式clone方法
Std4=(Student)Std3.clone();
//第四种方式
@SuppressWarnings("resource")
ObjectInputStream in = new ObjectInputStream(new FileInputStream("data.obj"));
Std5 = (Student) in.readObject();
} catch (IllegalArgumentException| InvocationTargetException| CloneNotSupportedException|InstantiationException | IllegalAccessException |ClassNotFoundException |IOException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
在Java开发特别是数据库开发中,经常会用到Class.forName( )这个方法。通过查询Java Documentation我们会发现使用Class.forName( )静态方法的目的是为了动态加载类。在加载完成后,一般还要调用Class下的newInstance( )静态方法来实例化对象以便操作。因此,单单使用Class.forName( )是动态加载类是没有用的,其最终目的是为了实例化对象。
有数据库开发经验朋友会发现,为什么在我们加载数据库驱动包的时候有的却没有调用newInstance( )方法呢?即有的jdbc连接数据库的写法里是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),为什么会有这两种写法呢?
刚才提到,Class.forName("");的作用是要求JVM查找并加载指定的类,如果在类中有静态初始化器的话,JVM必然会执行该类的静态代码段。而在JDBC规范中明确要求这个Driver类必须向DriverManager注册自己,即任何一个JDBC Driver的Driver类的代码都必须类似如下:
public class MyJDBCDriver implements Driver {
static {
DriverManager.registerDriver(new MyJDBCDriver());
}
}
既然在静态初始化器的中已经进行了注册,所以我们在使用JDBC时只需要Class.forName(XXX.XXX);就可以了。Java中工厂模式经常使用newInstance()方法来创建对象,因此从为什么要使用工厂模式上可以找到具体答案。
class c = Class.forName("Example");
factory = (ExampleInterface)c.newInstance();
其中ExampleInterface是Example的接口,可以写成如下形式:
String className = "Example";
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
进一步可以写成如下形式:
String className = readfromXMlConfig;//从xml 配置文件中获得字符串
class c = Class.forName(className);
factory = (ExampleInterface)c.newInstance();
上面代码已经不存在Example的类名称,它的优点是,无论Example类怎么变化,上述代码不变,甚至可以更换Example的兄弟类Example2 , Example3 , Example4……,只要他们继承ExampleInterface就可以。从JVM的角度看,我们使用关键字new创建一个类的时候,这个类可以没有被加载。但是使用newInstance()方法的时候,就必须保证:1、这个 类已经加载;2、这个类已经连接了。而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载 java API的那个加载器。
现在可以看出,newInstance()实际上是把new这个方式分解为两步,即首先调用Class加载方法加载某个类,然后实例化。 这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了一种降耦的手段。
最后用最简单的描述来区分new关键字和newInstance()方法的区别:
newInstance: 弱类型。低效率。只能调用无参构造。
new: 强类型。相对高效。能调用任何public构造。