關(guān)于java深拷貝的疑問
問題描述
下面這段代碼中,為什么
//對(duì)引用的對(duì)象也進(jìn)行復(fù)制o.p=(Professor)p.clone();
就能夠?qū)崿F(xiàn)深拷貝呢?
class Professor implements Cloneable { String name; int age; Professor(String name,int age) { this.name=name; this.age=age; } public Object clone() { Object o=null; try { o=super.clone(); } catch(CloneNotSupportedException e) { System.out.println(e.toString()); } return o; } } public class Student implements Cloneable { String name; int age; Professor p; Student(String name,int age,Professor p) { this.name=name; this.age=age; this.p=p; } public Object clone() { Student o=null; try { o=(Student)super.clone(); } catch(CloneNotSupportedException e) { System.out.println(e.toString()); } //對(duì)引用的對(duì)象也進(jìn)行復(fù)制 o.p=(Professor)p.clone(); return o; } public static void main(String[] args) { Professor p=new Professor('wangwu',50); Student s1=new Student('zhangsan',18,p); Student s2=(Student)s1.clone(); s2.p.name='lisi'; s2.p.age=30; //學(xué)生1的教授不 改變。 System.out.println('name='+s1.p.name+','+'age='+s1.p.age); System.out.println('name='+s2.p.name+','+'age='+s2.p.age); } }
問題解答
回答1:這只是表面上看起來是“深拷貝”, 實(shí)際上Student、Professor都沒有實(shí)現(xiàn)深拷貝。
你在main方法增加幾個(gè)輸出:
Professor p=new Professor('wangwu',50);Student s1=new Student('zhangsan',18,p);Student s2=(Student)s1.clone();System.out.println(s1.p == s2.p); //falseSystem.out.println(s1.name == s2.name); // trueSystem.out.println(s1.p.name == s2.p.name); //trues2.p.name='lisi';s2.p.age=30;//學(xué)生1的教授不 改變。System.out.println('name='+s1.p.name+','+'age='+s1.p.age);System.out.println('name='+s2.p.name+','+'age='+s2.p.age);System.out.println(s1.name == s2.name); //trueSystem.out.println(s1.p.name == s2.p.name); //false
即可看到,s1,s2的name仍然是'==', 在未set p.name時(shí)其name也為'==', 所以說都沒有實(shí)現(xiàn)深拷貝。當(dāng)你set s2.p.name時(shí),s2.p.name指向了另一個(gè)字符串常量的地址,所以(s1.p.name == s2.p.name); //false
回答2:你這個(gè)是淺復(fù)制,只能復(fù)制基本的數(shù)據(jù)類型,要復(fù)制對(duì)象成員變量,還需要調(diào)用該成員變量的clone方法,我是這么理解的,多次淺復(fù)制實(shí)現(xiàn)深復(fù)制
回答3:并不是深拷貝, 你可以試一下clone得到的professor.name與原來的professor.name是==的, 說明String還是引用的原來的
回答4:jdk clone方法的默認(rèn)實(shí)現(xiàn)都是value copy,對(duì)于基本類型,就是把copy值。對(duì)于引用,就是copy引用所指向的地址。
所以如果沒有o.p=(Professor)p.clone();這段代碼,那么原對(duì)象和clone對(duì)象的p,引用的都是同一個(gè)Professor對(duì)象,也就是淺copy。
相關(guān)文章:
1. golang - 用IDE看docker源碼時(shí)的小問題2. html - css氣泡,實(shí)現(xiàn)“倒三角(不知道算不算三角了)”可透明的。3. phpstudy8.1沒集成mysql-front4. java - 如何點(diǎn)擊按鈕,重新運(yùn)行(我是初學(xué)者)?5. 運(yùn)行python程序時(shí)出現(xiàn)“應(yīng)用程序發(fā)生異常”的內(nèi)存錯(cuò)誤?6. 我何時(shí)應(yīng)該在Java中使用JFrame.add(component)和JFrame.getContentPane()。add(component)7. 利用IPMI遠(yuǎn)程安裝centos報(bào)錯(cuò)!8. 前端 - @media query 使用出現(xiàn)的問題?9. html5和Flash對(duì)抗是什么情況?10. javascript - 請(qǐng)指條明路,angular的$event,在select中卻是undefined?

網(wǎng)公網(wǎng)安備