一文搞懂Java中创建对象的5种方式+实例分析

前言
说起Java手动创建对象一共有多少种方式这个问题,是自己最近一次面试的时候被问到的。当时自己只知道new和Class.newInstance这两种方式,现在学习汇总一下:
本文将介绍5种方式来创建一个java对象:

1、new关键字
2、Class.newInstance()
3、Constructor.newInstance()
4、clone()
5、反序列化

1、new关键字
这是我们最常见的也是最简单的创建对象的方式,通过这种方式我们还可以调用任意的构造器(无参的和有参的)。

public class Main {
    public static void main(String[] args) {
        Person person1 = new Person();
        Person person2 = new Person("gzc", 18);
    }
}

2、Class.newInstance()
这是我们运用反射创建对象时最常用的方法。Class类的newInstance()使用的是类的public无参构造器。因此也就是说使用此方法创建对象的前提是必须有public的无参构造器才行,否则报错如下:

// 没无参构造器报错信息
Caused by: java.lang.NoSuchMethodException: com.fsx.bean.Person.<init>()
	at java.lang.Class.getConstructor0(Class.java:3082)
	at java.lang.Class.newInstance(Class.java:412)
	... 1 more

// 无参构造器不是public的报错信息
Exception in thread "main" java.lang.IllegalAccessException: Class com.fsx.maintest.Main can not access a member of class com.fsx.bean.Person with modifiers "private"
	at sun.reflect.Reflection.ensureMemberAccess(Reflection.java:102)
	at java.lang.Class.newInstance(Class.java:436)
	at com.fsx.maintest.Main.main(Main.java:13)

正确使用方式如下:

public class Main {

    public static void main(String[] args) throws Exception {
        Person person = Person.class.newInstance();
        System.out.println(person); // Person{name='null', age=null}
    }
}

3、Constructor.newInstance()
本方法和Class类的newInstance方法很像,但是比它强大很多。
java.lang.relect.Constructor类里也有一个newInstance方法可以创建对象。我们可以通过这个newInstance方法调用有参(不再必须是无参)的和私有的构造函数(不再必须是public)。

public class Main {

    public static void main(String[] args) throws Exception {
        // 包括public的和非public的,当然也包括private的
        Constructor<?>[] declaredConstructors = Person.class.getDeclaredConstructors();
        // 只返回public的~~~~~~(返回结果是上面的子集)
        Constructor<?>[] constructors = Person.class.getConstructors();


        Constructor<?> noArgsConstructor = declaredConstructors[0];
        Constructor<?> haveArgsConstructor = declaredConstructors[1];

        noArgsConstructor.setAccessible(true); // 非public的构造必须设置true才能用于创建实例
        Object person1 = noArgsConstructor.newInstance();
        Object person2 = declaredConstructors[1].newInstance("gzc", 18);

        System.out.println(person1);
        System.out.println(person2);
    }

}
//输出结果如下:
Person{name='null', age=null}
Person{name='gzc', age=18}

4、clone()
无论何时我们调用一个对象的clone方法,JVM就会创建一个新的对象,将前面的对象的内容全部拷贝进去,用clone方法创建对象并不会调用任何构造函数
要使用clone方法,我们必须先实现Cloneable接口并复写Object的clone方法(因为Object的这个方法是protected的,你若不复写,外部也调用不了呀)。

public class Person implements Cloneable {
	...
	// 访问权限写为public,并且返回值写为person
    @Override
    public Person clone() throws CloneNotSupportedException {
        return (Person) super.clone();
    }
    ...
}

public class Main {

    public static void main(String[] args) throws Exception {
        Person person = new Person("gzc", 18);
        Object clone = person.clone();

        System.out.println(person);
        System.out.println(clone);
        System.out.println(person == clone); //false
    }

}
//输出结果如下:
Person{name='gzc', age=18}
Person{name='gzc', age=18}
false   //输出false就说明该方法克隆出的是一个全新的对象。

5、反序列化
当我们序列化和反序列化一个对象,JVM会给我们创建一个单独的对象,在反序列化时,JVM创建对象并不会调用任何构造函数。为了反序列化一个对象,我们需要让我们的类实现Serializable接口。

public class Main {

    public static void main(String[] args) throws Exception {
        Person person = new Person("gzc", 18);
        byte[] bytes = SerializationUtils.serialize(person);

        // 字节数组:可以来自网络、可以来自文件(本处直接本地模拟)
        Object deserPerson = SerializationUtils.deserialize(bytes);
        System.out.println(person);
        System.out.println(deserPerson);
        System.out.println(person == deserPerson);
    }

}
//输出结果如下:
Person{name='gzc', age=18}
Person{name='gzc', age=18}
false
//注意:JDK序列化、反序列化特别特别耗内存。

6、题外话
有了上面的学习其实又可以衍生出两个知识点:
(1)Java创建实例对象是不是必须要通过构造函数?
答案已经很明显了:Java创建实例对象,并不一定必须要调用构造器的,总结如下:
在这里插入图片描述

(2)上述两种newInstance()的区别?
①Class类位于java的lang包中,而Constructor是java反射机制的一部分;
②Class类的newInstance只能触发无参的构造方法创建对象,而构造器类的newInstance能触发任意参数的构造方法来创建对象。
③Class类的newInstance需要其构造方法是public的或者对调用方法可见的,而构造器类的newInstance可以在特定环境下调用私有构造方法来创建对象。
④Class类的newInstance抛出类构造函数的异常,而构造器类的newInstance包装了一个InvocationTargetException异常。
⑤ Class类本质上调用了反射包Constructor中无参的newInstance方法,捕获了InvocationTargetException,将构造器本身的异常抛出。

对了,还有一个小tips:其实还有一个库Objenesis,它也能不使用构造器来创建一个实例。Spring的ObjenesisCglibAopProxy就是依赖于Objenesis这个库的~ 比较冷门的一个知识点,了解了解即可~

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python是一门支持面向对象编程的语言,它通过类(Class)和对象(Object)的概念实现了这一特性。以下是关于Python面向对象编程的一般理解: 1. **类(Class)**:类是一模板或蓝图,用于创建具有相似属性和行为的对象。类定义了数据成员(如变量)和方法(函数),它们描述了对象的状态和行为。 2. **对象(Object)**:对象是类的实例,它是现实世界的一个实体。每个对象都具有特定的属性值,并能够执行其类定义的方法。 3. **属性(Attribute)**:类的变量就是对象的属性,它们可以存储数据。比如,一个人类对象可能有姓名、年龄这样的属性。 4. **方法(Method)**:类的函数是对象的行为,例如获取信息(getter)、设置信息(setter)、执行动作等。比如,人的类可能会有一个方法“说话”(speak)。 5. **封装(Encapsulation)**:将数据和操作数据的代码打包成类,隐藏实现细节,仅对外提供接口访问,保护数据的安全性。 6. **继承(Inheritance)**:子类可以继承父类的属性和方法,通过"IS-A"关系实现代码复用。子类可以添加新的属性和方法,也可以覆盖或扩展父类的方法。 7. **多态(Polymorphism)**:同一行为可以在不同的对象上表现出不同的形式,包括静态多态(方法重载)和动态多态(方法重写或虚函数)。 8. **构造函数(Constructor)**:特殊的方法,当创建新对象时自动执行,通常用于初始化对象的属性。 9. **析构函数(Destructor)**:特殊的方法,在对象生命周期结束时自动执行,用于清理资源。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值