用Java创建对象的5种不同方法

createh53周前 (12-09)技术教程17

作为Java开发人员,我们通常每天创建许多对象,但是我们始终使用依赖管理系统(例如Spring)来创建这些对象。但是,还有更多创建对象的方法,将在本文中进行研究。

Java中共有五种创建对象的方法,下面将通过示例说明这些方法,然后介绍创建对象的行的字节码。

执行最后给出的程序,将会看到方法1、2、3使用构造函数创建对象,而4、5没有调用构造函数创建对象。

1.使用新关键字

这就是日常的方法

Employee emp1 = new Employee();
 0: new           #19          // class org/programming/mitra/exercises/Employee
 3: dup
 4: invokespecial #21          // Method org/programming/mitra/exercises/Employee."":()V

2.使用Class类的newInstance()方法

以使用Class类的newInstance()方法创建一个对象。此newInstance()方法调用no-arg构造函数来创建对象。

可以通过newInstance()通过以下方式创建对象:

Employee emp2 = (Employee) Class.forName("org.programming.mitra.exercises.Employee").newInstance();

或者

Employee emp2 = Employee.class.newInstance();
51: invokevirtual    #70    // Method java/lang/Class.newInstance:()Ljava/lang/Object;

3.使用构造方法类的newInstance()方法

与Class类的newInstance()方法类似,java.lang.reflect.Constructor类中有一个newInstance()方法可用于创建对象。我们还可以使用此newInstance()方法来调用参数化构造函数和私有构造函数。

Constructor<Employee> constructor =Employee.class.getConstructor();
Employee emp3 = constructor.newInstance();
111: invokevirtual  #80  // Method java/lang/reflect/Constructor.newInstance:([Ljava/lang/Object;)Ljava/lang/Object;

这两种newInstance()方法都被称为创建对象的反射方式。实际上,Class类的newInstance()方法在内部使用了Constructor类的newInstance()方法。这就是为什么最好使用后一种方法,并在Spring,Hibernate,Struts等不同框架中使用后一种方法的原因。

4.使用clone()方法:

每当我们在任何对象上调用clone()时,JVM实际上都会为我们创建一个新对象,并将先前对象的所有内容复制到其中。使用clone方法创建对象不会调用任何构造函数。

要在对象上使用clone()方法,我们需要实现Cloneable并在其中定义clone()方法。

Employee emp4 = (Employee) emp3.clone();
162: invokevirtual #87  // Method org/programming/mitra/exercises/Employee.clone ()Ljava/lang/Object;

这种方式肯定也有其缺点的,但是在对象完全满足Java克隆的强制条件之前,它仍然是创建任何对象的副本的最流行和最简单的方法。

5.使用反序列化:

每当我们序列化和反序列化一个对象时,JVM都会为我们创建一个单独的对象。在反序列化中,JVM不使用任何构造函数来创建对象。

为了反序列化一个对象,我们需要在我们的类中实现一个Serializable接口。

ObjectInputStream in = new ObjectInputStream(new FileInputStream("data.obj"));
Employee emp5 = (Employee) in.readObject();
261: invokevirtual  #118   // Method java/io/ObjectInputStream.readObject:()Ljava/lang/Object;

正如我们在上面的字节码片段中看到的,invokevirtual 除了第一个方法被转换为两个调用之外,所有四个方法都被调用并转换为 (这些方法直接处理对象的创建):一个是新的,另一个是invokespecial (调用构造函数)。

考虑一个Employee类,将为其创建对象:

class Employee implements Cloneable, Serializable {
    private static final long serialVersionUID = 1L;
    private String name;
    public Employee() {
        System.out.println("Employee Constructor Called...");
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        return result;
    }
    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;
        if (obj == null)
            return false;
        if (getClass() != obj.getClass())
            return false;
        Employee other = (Employee) obj;
        if (name == null) {
            if (other.name != null)
                return false;
        } else if (!name.equals(other.name))
            return false;
        return true;
    }
    @Override
    public String toString() {
        return "Employee [name=" + name + "]";
    }
    @Override
    public Object clone() {
        Object obj = null;
        try {
            obj = super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return obj;
    }
}

在下面的Java程序中,将用所有5种方式创建Employee对象。

public class ObjectCreation {
    public static void main(String... args) throws Exception {
        // 使用新关键字
        Employee emp1 = new Employee();
        emp1.setName("xiaoming");
        System.out.println(emp1 + ", hashcode : " + emp1.hashCode());
        // 通过使用Class类的newInstance()方法
        Employee emp2 = (Employee) Class.forName("org.programming.mitra.exercises.Employee")
                               .newInstance();
        //或者我们可以简单地做到这一点
        //员工emp2 = Employee.class.newInstance();
        emp2.setName("xiaofang");
        System.out.println(emp2 + ", hashcode : " + emp2.hashCode());
        // 通过使用Constructor类的newInstance()方法
        Constructor<Employee> constructor = Employee.class.getConstructor();
        Employee emp3 = constructor.newInstance();
        emp3.setName("xiaohong");
        System.out.println(emp3 + ", hashcode : " + emp3.hashCode());
        // 过使用clone()方法
        Employee emp4 = (Employee) emp3.clone();
        emp4.setName("xiaoai");
        System.out.println(emp4 + ", hashcode : " + emp4.hashCode());
         //通过反序列化
        //序列化
        ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("data.obj"));
        out.writeObject(emp4);
        out.close();
       //反序列化
        ObjectInputStream in = new ObjectInputStream(new FileInputStream("data.obj"));
        Employee emp5 = (Employee) in.readObject();
        in.close();
        emp5.setName("zhazha");
        System.out.println(emp5 + ", hashcode : " + emp5.hashCode());
    }
}

该程序将给出以下输出:

Employee Constructor Called...
Employee [name=xiaoming], hashcode : -1968815046
Employee Constructor Called...
Employee [name=xiaofang], hashcode : 78970652
Employee Constructor Called...
Employee [name=xiaohong], hashcode : -1641292792
Employee [name=xiaoai], hashcode : 2051657
Employee [name=zhazha], hashcode : 63313419

喜欢希望您关注下呗。

相关文章

这篇Java基础快速入门学习教程,让我至少少走3个月弯路

作为刚刚接触java的小白,今天这篇文章的目的是帮助大家快速入门java,少走弯路如何快速入门Java?一、作为刚接触Java的新手,先了解Java的基础。Java 基本语法、Java 平台应用、Ja...

Java中super关键字怎样使用?

当子类重写父类的方法后,子类对象将无法直接访问父类被重写的方法。为了解决这个问题,在Java中专门提供了一个super关键字来访问父类的成员,例如访问父类的成员变量、成员方法和构造方法。下面分两种情况...

我的世界:关于鼠标中键,你不知道的11种用法!白玩了这么多年mc

问:如何在原版mc下,一键把箱子A中所有的物品,复制到箱子B中?这个问题很难么?这是一篇看完之后,一定会让你觉得这么多年《我的世界》算是白玩了!因为本期我们要聊的是基于一些快捷键下的Minecraft...

3分钟教你如何自学Java

我们学习完java的基础后,会感到自己学习到了很多的知识,但是这些知识都是非常的抽象的。虽然学的挺多的,但我们更加需要的是操作,将这些知识通过练习与理解,让它变成自己的一部分。 刚开始学习时,课本上...

实操讲解Java中反射的机制,使用,案例;再看不懂神仙也救不了你

Reflection 是 Java 程序开发语言的特征之一,它允许运行中的 Java 程序对自身进行检查,或者说“自审”,并能直接操作程序的内部属性。例如,使用它能获得 Java 类中各成员的名称并显...