Java?中的?clone(?)?和?new哪個(gè)效率更高
對(duì)象創(chuàng)建的幾種方法:
- 使用new關(guān)鍵字
- 使用clone方法
- 反射機(jī)制
- 反序列化
以上四種都可以產(chǎn)生java對(duì)象
- 1,3都會(huì)明確的顯式的調(diào)用構(gòu)造函數(shù)
- 2是在內(nèi)存上對(duì)已有對(duì)象的影印 所以不會(huì)調(diào)用構(gòu)造函數(shù)
- 4是從文件中還原類(lèi)的對(duì)象 也不會(huì)調(diào)用構(gòu)造函數(shù)
何為clone()?
- 拷貝對(duì)象返回的是一個(gè)新的對(duì)象,而不是一個(gè)對(duì)象的引用地址;
- 拷貝對(duì)象已經(jīng)包含原來(lái)對(duì)象的信息,而不是對(duì)象的初始信息,即每次拷貝動(dòng)作不是針對(duì)一個(gè)全新對(duì)象的創(chuàng)建。
clone()和new那個(gè)更快?
利用clone,在內(nèi)存中進(jìn)行數(shù)據(jù)塊的拷貝,復(fù)制已有的對(duì)象,也是生成對(duì)象的一種方式。前提是類(lèi)實(shí)現(xiàn)Cloneable接口,Cloneable接口沒(méi)有任何方法,是一個(gè)空接口,也可以稱這樣的接口為標(biāo)志接口,只有實(shí)現(xiàn)了該接口,才會(huì)支持clone操作。有的人也許會(huì)問(wèn)了,java中的對(duì)象都有一個(gè)默認(rèn)的父類(lèi)Object。
Object中有一個(gè)clone方法,為什么還必須要實(shí)現(xiàn)Cloneable接口呢,這就是cloneable接口這個(gè)標(biāo)志接口的意義,只有實(shí)現(xiàn)了這個(gè)接口才能實(shí)現(xiàn)復(fù)制操作,因?yàn)閖vm在復(fù)制對(duì)象的時(shí)候,會(huì)檢查對(duì)象的類(lèi)是否實(shí)現(xiàn)了Cloneable這個(gè)接口,如果沒(méi)有實(shí)現(xiàn),則會(huì)報(bào)CloneNotSupportedException異常。類(lèi)似這樣的接口還有Serializable接口、RandomAccess接口等。
還有值得一提的是在執(zhí)行clone操作的時(shí)候,不會(huì)調(diào)用構(gòu)造函數(shù)。還有clone操作還會(huì)面臨深拷貝和淺拷貝的問(wèn)題。關(guān)于這方面的問(wèn)題,網(wǎng)上有很多的相關(guān)知識(shí)了,不再累述了。由于通過(guò)復(fù)制操作得到對(duì)象不需要調(diào)用構(gòu)造函數(shù),只是內(nèi)存中的數(shù)據(jù)塊的拷貝,那是不是拷貝對(duì)象的效率是不是一定會(huì)比new的時(shí)候的快。
答案:不是。顯然jvm的開(kāi)發(fā)者也意識(shí)到通過(guò)new方式來(lái)生成對(duì)象占據(jù)了開(kāi)發(fā)者生成對(duì)象的絕大部分,所以對(duì)于利用new操作生成對(duì)象進(jìn)行了優(yōu)化。
例如:
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));
}
}
打印結(jié)果:

new完勝clone,真的是這樣嗎?
下面在構(gòu)造函數(shù)里做點(diǎn)簡(jiǎn)單的事情,例如字符串截取試試。只是修改Bean,其他不變?cè)倏创蛴?/p>
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();
}
}

結(jié)論:輕量級(jí)的對(duì)象可以使用new,其他對(duì)象可以使用clone。
到此這篇關(guān)于Java 中的 clone( ) 和 new哪個(gè)效率更高的文章就介紹到這了,更多相關(guān)Java clone( ) 和 new內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java參數(shù)傳遞及值傳遞實(shí)現(xiàn)原理詳解
這篇文章主要介紹了Java參數(shù)傳遞及值傳遞實(shí)現(xiàn)原理詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-08-08
Java中@DS+@Transactional注解切換數(shù)據(jù)源失效解決方案
本文主要介紹了@DS+@Transactional注解切換數(shù)據(jù)源失效解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-06-06
Spring Security單項(xiàng)目權(quán)限設(shè)計(jì)過(guò)程解析
這篇文章主要介紹了Spring Security單項(xiàng)目權(quán)限設(shè)計(jì)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-11-11
Maven項(xiàng)目改為spring boot項(xiàng)目的方法圖解
這篇文章主要介紹了Maven項(xiàng)目改為spring boot項(xiàng)目的方法圖解 ,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-09-09
maven項(xiàng)目在實(shí)踐中的構(gòu)建管理之路的方法
這篇文章主要介紹了maven項(xiàng)目在實(shí)踐中的構(gòu)建管理之路的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-05-05

