Java中new与clone操作对象的比较方法举例
作者:Seal^_^
1、new操作符创建对象的过程
new操作符在Java中用于创建对象,并执行以下步骤:
- 分配内存空间: 根据类型确定所需内存大小。
- 调用构造函数: 初始化对象的各个域。
- 返回对象引用: 可以将引用发布到外部,使得对象可以被使用。
示例代码:
MyClass obj = new MyClass();
new MyClass()创建了一个MyClass类型的对象并分配了内存空间。
2、clone方法创建对象的过程
- 分配内存与初始化:
- clone方法和new操作符都会分配内存。但是,clone方法分配的内存大小与原对象相同。
- 不同于new操作符,clone方法不会调用构造函数来初始化对象的域。而是使用原对象的值填充新对象的域。这称为 浅拷贝(shallow copy)。
- 浅拷贝与深拷贝的区别:
- 如果对象的域中引用了其他对象(例如数组或其他类的实例),clone方法只会复制引用,而不会复制引用指向的对象本身。
- 这就是浅拷贝和深拷贝的区别:浅拷贝复制引用,深拷贝则会递归复制引用的对象。
- 返回新对象:
- 当clone方法执行完毕后,将会创建一个新的对象,其值与原对象相同(但内存地址不同)。
- 我们可以将新对象的引用用于外部。
示例代码:
MyClass original = new MyClass(); MyClass cloned = (MyClass) original.clone();
3、总结
1、new操作符:分配内存并调用构造函数初始化对象。
2、clone方法:分配内存并使用原对象的值来填充新对象的域(浅拷贝)。
虽然new和clone都可以用来创建对象,但它们的使用场景和原理是不同的。在实际编程中,我们需要根据具体的需求来选择使用哪种方式。
附:clone()和new那个更快?
利用clone,在内存中进行数据块的拷贝,复制已有的对象,也是生成对象的一种方式。前提是类实现Cloneable接口,Cloneable接口没有任何方法,是一个空接口,也可以称这样的接口为标志接口,只有实现了该接口,才会支持clone操作。有的人也许会问了,java中的对象都有一个默认的父类Object。
Object中有一个clone方法,为什么还必须要实现Cloneable接口呢,这就是cloneable接口这个标志接口的意义,只有实现了这个接口才能实现复制操作,因为jvm在复制对象的时候,会检查对象的类是否实现了Cloneable这个接口,如果没有实现,则会报CloneNotSupportedException异常。类似这样的接口还有Serializable接口、RandomAccess接口等。
还有值得一提的是在执行clone操作的时候,不会调用构造函数。还有clone操作还会面临深拷贝和浅拷贝的问题。关于这方面的问题,网上有很多的相关知识了,不再累述了。由于通过复制操作得到对象不需要调用构造函数,只是内存中的数据块的拷贝,那是不是拷贝对象的效率是不是一定会比new的时候的快。
答案:不是。显然jvm的开发者也意识到通过new方式来生成对象占据了开发者生成对象的绝大部分,所以对于利用new操作生成对象进行了优化。
例如:
package com.miivii.javalib; public class Bean implements Cloneable { private String name; public Bean(String name) { this.name = name; } @Override protected Bean clone() throws CloneNotSupportedException { return (Bean) super.clone(); } }
package com.miivii.javalib; public class TestClass { private static final int COUNT = 10000 * 1000; public static void main(String[] args) throws CloneNotSupportedException { long s1 = System.currentTimeMillis(); for (int i = 0; i < COUNT; i++) { Bean bean = new Bean("ylWang"); } long s2 = System.currentTimeMillis(); Bean bean = new Bean("ylWang"); for (int i = 0; i < COUNT; i++) { Bean b = bean.clone(); } long s3 = System.currentTimeMillis(); System.out.println("new = " + (s2 - s1)); System.out.println("clone = " + (s3 - s2)); } }
打印结果:
new完胜clone,真的是这样吗?
下面在构造函数里做点简单的事情,例如字符串截取试试。只是修改Bean,其他不变再看打印
package com.miivii.javalib; public class Bean implements Cloneable { private String name; private String firstSign;//获取名字首字母 public Bean(String name) { this.name = name; if (name.length() != 0) { firstSign = name.substring(0, 1); firstSign += "abc"; } } @Override protected Bean clone() throws CloneNotSupportedException { return (Bean) super.clone(); } }
结论:轻量级的对象可以使用new,其他对象可以使用clone。
到此这篇关于Java中new与clone操作对象的比较方法的文章就介绍到这了,更多相关Java new与clone操作对象比较内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!