Java创建对象有几种方式?
使用New关键字、使用Class类的newInstance方法、使用Constructor类的newInstance方法、使用Clone方法、使用反序列化。
-
使用new关键字:
使用 new 关键字创建对象,实际上是做了两个工作,
一是在内存中开辟空间,二是初始化对象。比如:
Student student = new Student();
-
反射是对于任意一个正在运行的类,都能动态获取到他的属性和方法。
反射创建对象分为两种方式,一是使用Class类的new Instance() 方法,二是使用Constructor类的new Instatnce() 方法。使用Class类的newInstance方法:
newInstance方法调用无参的构造器创建对象,
比如:Student student2 = (Student)Class.forName("根路径.Student").newInstance();
或者:Student stu = Student.class.newInstance();
Class.newInstance() 只能够调用无参的构造函数,即默认的构造函数;使用Constructor类的newInstance方法:
java.lang.relect.Constructor类中的newInstance方法可以创建对象。
我们可以通过这个newInstance方法调用有参数的和私有的构造函数。
如:Constructor<Student> constructor = Student.class.getInstance(); Student stu = constructor.newInstance();
Constructor.newInstance() 可以根据传入的参数,调用任意构造构造函数。 -
使用Clone的方法:
无论何时我们调用一个对象的clone方法,JVM就会创建一个新的对象,将前面的对象的内容全部拷贝进去,
用clone方法创建对象并不会调用任何构造函数。
要使用clone方法,我们必须先实现Cloneable接口并实现其定义的clone方法。
如:
Student stu2 = <Student>stu.clone();
(这也是原型模式的应用。) -
使用反序列化:
当我们序列化和反序列化一个对象,JVM会给我们创建一个单独的对象,
在反序列化时,JVM创建对象并不会调用任何构造函数。
为了反序列化一个对象,我们需要让我们的类实现Serializable接口,然后在使用new ObjectInputStream().readObject() 来创建对象。
如:
ObjectInputStream in = new ObjectInputStream (new FileInputStream("data.obj"));
Student stu3 = (Student)in.readObject();
从上面的例子可以看出来,除了使用new关键字之外的其他方法全部都是转变为invokevirtual(创建对象的直接方法),使用被new的方式转变为两个调用,new和invokespecial(构造函数调用)。new和invokespecial两种newInstance方法的区别:
☆ Class类位于java的lang包中,而构造器类是java反射机制的一部分。
☆ Class类的newInstance只能触发无参构造方法创建对象,而构造器类的newInstance能触发有参或者任意参数的构造方法来创建对象。
☆ Class类的newInstance需要构造方法是public或者对调用方法可见,构造器类的newInstance可在特定环境下调用私有构造方法创建对象。
☆ Class类的newInstance抛出类构造函数的异常,而构造器类的newInstance包装了一个InvocationTargetException异常。Class类本质上调用了反射包构造器类中无参数的newInstance方法,捕获了InvocationTargetException将构造器本身的异常抛出。
扩展:深拷贝和浅拷贝的区别是什么?
浅拷贝只是增加了一个指针指向已存在的内存地址,仅仅是指向被复制的内存地址,
如果原地址发生改变,那么浅复制出来的对象也会相应的改变。
深拷贝是增加了一个指针并且申请了一个新的内存,使这个增加的指针指向这个新的内存。