相关阅读
【小家java】java5新特性(简述十大新特性) 重要一跃
【小家java】java6新特性(简述十大新特性) 鸡肋升级
【小家java】java7新特性(简述八大新特性) 不温不火
【小家java】java8新特性(简述十大新特性) 饱受赞誉
【小家java】java9新特性(简述十大新特性) 褒贬不一
【小家java】java10新特性(简述十大新特性) 小步迭代
【小家java】java11新特性(简述八大新特性) 首个重磅LTS版本
作为一个Java开发者,一种面向对象的语言,我们每天都创建很多对象。但后续我们开发中,采用了spring的依赖管理系统,我们就很少自己去创建对象了,全部交给容器去托管,那么本篇文章回源塑本,讲述一下java中能够创建一个对象的5中方法。
本文最大的特色是,我不仅给出案例,还给出对应的字节码解释,从底层来解释现象
1.使用new关键字
这是最常见也是最简单的创建对象的方式了。通过这种方式,我们可以调用任意的构造函数(无参的和带参数的)。
Employee emp1 = new Employee();
字节码:
0: new #19 // class com/fsx/demo/Employee 3: dup 4: invokespecial #21 // Method com/fsx/demo/Employee."":()V
2.使用Class类的newInstance方法
这个方法创建对象其实我们用得也比较多,but,这个newInstance方法调用无参的构造函数创建对象。所以类必须有public无参构造函数才行
Employee emp2 = (Employee) Class.forName("com.fsx.demo.mployee").newInstance(); 或者 Employee emp2 = Employee.class.newInstance();
字节码:
51: invokevirtual #70 // Method java/lang/Class.newInstance:()Ljava/lang/Object;
3.使用Constructor类的newInstance方法
和Class类的newInstance方法很像。但是它可以调用任意构造函数创建对象,包括私有的。(所以即使你私有了构造函数,spring还是可以给你创建对象)
事实上Class的newInstance方法内部调用Constructor的newInstance方法。这也是众多框架,如Spring、Hibernate、Struts等直接使用后者的原因
4.使用clone方法
无论何时我们调用一个对象的clone方法,jvm就会创建一个新的对象,将前面对象的内容全部拷贝进去。用clone方法创建对象并不会调用任何构造函数。
备注:要使用clone方法,我们需要先实现Cloneable接口并实现其定义的clone方法。
Employee emp4 = (Employee) emp2.clone();
字节码:
162: invokevirtual #87 // Method com/fsx/demo//Employee.clone ()Ljava/lang/Object;
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;
总结:
我们从上面的字节码片段可以看到,除了第1个方法,其他4个方法全都转变为invokevirtual(创建对象的直接方法),第一个方法转变为两个调用,new和invokespecial(构造函数调用)。
直接给出是否调用了构造函数的结论:
- 使用new关键字 } → 调用了构造函数
- 使用Class类的newInstance方法 } → 调用了构造函数
- 使用Constructor类的newInstance方法 } → 调用了构造函数
- 使用clone方法 } → 没有调用构造函数
- 使用反序列化 } → 没有调用构造函数