Java创建对象有几种方式?

使用New关键字、使用Class类的newInstance方法、使用Constructor类的newInstance方法、使用Clone方法、使用反序列化。

  1. 使用new关键字:

    使用 new 关键字创建对象,实际上是做了两个工作,
    一是在内存中开辟空间,二是初始化对象。

    比如:Student student = new Student();

  2. 反射是对于任意一个正在运行的类,都能动态获取到他的属性和方法。
    反射创建对象分为两种方式,一是使用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() 可以根据传入的参数,调用任意构造构造函数。

  3. 使用Clone的方法:

    无论何时我们调用一个对象的clone方法,JVM就会创建一个新的对象,将前面的对象的内容全部拷贝进去,

    用clone方法创建对象并不会调用任何构造函数。

    要使用clone方法,我们必须先实现Cloneable接口并实现其定义的clone方法。

    如:Student stu2 = <Student>stu.clone();(这也是原型模式的应用。)

  4. 使用反序列化:

    当我们序列化和反序列化一个对象,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将构造器本身的异常抛出。

扩展:深拷贝和浅拷贝的区别是什么?
浅拷贝只是增加了一个指针指向已存在的内存地址,仅仅是指向被复制的内存地址,
如果原地址发生改变,那么浅复制出来的对象也会相应的改变。
深拷贝是增加了一个指针并且申请了一个新的内存,使这个增加的指针指向这个新的内存。