在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

什么是深拷貝和淺拷貝

科技綠洲 ? 來源:Java技術(shù)指北 ? 作者:Java技術(shù)指北 ? 2023-10-13 11:29 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

1、創(chuàng)建對(duì)象的5種方式

①、通過 new 關(guān)鍵字

這是最常用的一種方式,通過 new 關(guān)鍵字調(diào)用類的有參或無參構(gòu)造方法來創(chuàng)建對(duì)象。比如 Object obj = new Object();

②、通過 Class 類的 newInstance() 方法

這種默認(rèn)是調(diào)用類的無參構(gòu)造方法創(chuàng)建對(duì)象。比如 Person p2 = (Person) Class.forName("com.ys.test.Person").newInstance();

③、通過 Constructor 類的 newInstance 方法

這和第二種方法類都是通過反射來實(shí)現(xiàn)。通過 java.lang.relect.Constructor 類的 newInstance() 方法指定某個(gè)構(gòu)造器來創(chuàng)建對(duì)象。

Person p3 = (Person) Person.class.getConstructors()[0].newInstance();

實(shí)際上第二種方法利用 Class 的 newInstance() 方法創(chuàng)建對(duì)象,其內(nèi)部調(diào)用還是 Constructor 的 newInstance() 方法。

④、利用 Clone 方法

Clone 是 Object 類中的一個(gè)方法,通過 對(duì)象A.clone() 方法會(huì)創(chuàng)建一個(gè)內(nèi)容和對(duì)象 A 一模一樣的對(duì)象 B,clone 克隆,顧名思義就是創(chuàng)建一個(gè)一模一樣的對(duì)象出來。

Person p4 = (Person) p3.clone();

⑤、反序列化

序列化是把堆內(nèi)存中的 Java 對(duì)象數(shù)據(jù),通過某種方式把對(duì)象存儲(chǔ)到磁盤文件中或者傳遞給其他網(wǎng)絡(luò)節(jié)點(diǎn)(在網(wǎng)絡(luò)上傳輸)。而反序列化則是把磁盤文件中的對(duì)象數(shù)據(jù)或者把網(wǎng)絡(luò)節(jié)點(diǎn)上的對(duì)象數(shù)據(jù),恢復(fù)成Java對(duì)象模型的過程。

2、Clone 方法

本篇博客我們講解的是 Java 的深拷貝和淺拷貝,其實(shí)現(xiàn)方式正是通過調(diào)用 Object 類的 clone() 方法來完成。在 Object.class 類中,源碼為:

protected native Object clone() throws CloneNotSupportedException;

這是一個(gè)用 native 關(guān)鍵字修飾的方法,關(guān)于native關(guān)鍵字有一篇博客專門有介紹,不理解也沒關(guān)系,只需要知道用 native 修飾的方法就是告訴操作系統(tǒng),這個(gè)方法我不實(shí)現(xiàn)了,讓操作系統(tǒng)去實(shí)現(xiàn)。具體怎么實(shí)現(xiàn)我們不需要了解,只需要知道 clone方法的作用就是復(fù)制對(duì)象,產(chǎn)生一個(gè)新的對(duì)象。那么這個(gè)新的對(duì)象和原對(duì)象是什么關(guān)系呢?

3、基本類型和引用類型

這里再給大家普及一個(gè)概念,在 Java 中基本類型和引用類型的區(qū)別。

在 Java 中數(shù)據(jù)類型可以分為兩大類:基本類型和引用類型。

基本類型也稱為值類型,分別是字符類型 char,布爾類型 boolean以及數(shù)值類型 byte、short、int、long、float、double。

引用類型則包括類、接口、數(shù)組、枚舉等。

Java 將內(nèi)存空間分為堆和棧。基本類型直接在棧中存儲(chǔ)數(shù)值,而引用類型是將引用放在棧中,實(shí)際存儲(chǔ)的值是放在堆中,通過棧中的引用指向堆中存放的數(shù)據(jù)。圖片 上圖定義的 a 和 b 都是基本類型,其值是直接存放在棧中的;而 c 和 d 是 String 聲明的,這是一個(gè)引用類型,引用地址是存放在 棧中,然后指向堆的內(nèi)存空間。

下面 d = c;這條語句表示將 c 的引用賦值給 d,那么 c 和 d 將指向同一塊堆內(nèi)存空間。

4、淺拷貝

我們看如下這段代碼:

package com.ys.test;

public class Person implements Cloneable{
    public String pname;
    public int page;
    public Address address;
    public Person() {}
    
    public Person(String pname,int page){
        this.pname = pname;
        this.page = page;
        this.address = new Address();
    }
    
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
    
    public void setAddress(String provices,String city ){
        address.setAddress(provices, city);
    }
    public void display(String name){
        System.out.println(name+":"+"pname=" + pname + ", page=" + page +","+ address);
    }

    public String getPname() {
        return pname;
    }

    public void setPname(String pname) {
        this.pname = pname;
    }

    public int getPage() {
        return page;
    }

    public void setPage(int page) {
        this.page = page;
    }
    
}

這是一個(gè)我們要進(jìn)行賦值的原始類 Person。下面我們產(chǎn)生一個(gè) Person 對(duì)象,并調(diào)用其 clone 方法復(fù)制一個(gè)新的對(duì)象。

注意:調(diào)用對(duì)象的 clone 方法,必須要讓類實(shí)現(xiàn) Cloneable 接口,并且覆寫 clone 方法。

測試:

@Test
public void testShallowClone() throws Exception{
    Person p1 = new Person("zhangsan",21);
    p1.setAddress("湖北省", "武漢市");
    Person p2 = (Person) p1.clone();
    System.out.println("p1:"+p1);
    System.out.println("p1.getPname:"+p1.getPname().hashCode());
    
    System.out.println("p2:"+p2);
    System.out.println("p2.getPname:"+p2.getPname().hashCode());
    
    p1.display("p1");
    p2.display("p2");
    p2.setAddress("湖北省", "荊州市");
    System.out.println("將復(fù)制之后的對(duì)象地址修改:");
    p1.display("p1");
    p2.display("p2");
}

打印結(jié)果為:圖片 首先看原始類 Person 實(shí)現(xiàn) Cloneable 接口,并且覆寫 clone 方法,它還有三個(gè)屬性,一個(gè)引用類型 String定義的 pname,一個(gè)基本類型 int定義的 page,還有一個(gè)引用類型 Address ,這是一個(gè)自定義類,這個(gè)類也包含兩個(gè)屬性 pprovices 和 city 。

接著看測試內(nèi)容,首先我們創(chuàng)建一個(gè)Person 類的對(duì)象 p1,其pname 為zhangsan,page為21,地址類 Address 兩個(gè)屬性為 湖北省和武漢市。接著我們調(diào)用 clone() 方法復(fù)制另一個(gè)對(duì)象 p2,接著打印這兩個(gè)對(duì)象的內(nèi)容。

從第 1 行和第 3 行打印結(jié)果:

p1:com.ys.test.Person@349319f9

p2:com.ys.test.Person@258e4566

可以看出這是兩個(gè)不同的對(duì)象。

從第 5 行和第 6 行打印的對(duì)象內(nèi)容看,原對(duì)象 p1 和克隆出來的對(duì)象 p2 內(nèi)容完全相同。

代碼中我們只是更改了克隆對(duì)象 p2 的屬性 Address 為湖北省荊州市(原對(duì)象 p1 是湖北省武漢市) ,但是從第 7 行和第 8 行打印結(jié)果來看,原對(duì)象 p1 和克隆對(duì)象 p2 的 Address 屬性都被修改了。

也就是說對(duì)象 Person 的屬性 Address,經(jīng)過 clone 之后,其實(shí)只是復(fù)制了其引用,他們指向的還是同一塊堆內(nèi)存空間,當(dāng)修改其中一個(gè)對(duì)象的屬性 Address,另一個(gè)也會(huì)跟著變化。圖片

淺拷貝:創(chuàng)建一個(gè)新對(duì)象,然后將當(dāng)前對(duì)象的非靜態(tài)字段復(fù)制到該新對(duì)象,如果字段是值類型的,那么對(duì)該字段執(zhí)行復(fù)制;如果該字段是引用類型的話,則復(fù)制引用但不復(fù)制引用的對(duì)象。因此,原始對(duì)象及其副本引用同一個(gè)對(duì)象。

5、深拷貝

弄清楚了淺拷貝,那么深拷貝就很容易理解了。

深拷貝:創(chuàng)建一個(gè)新對(duì)象,然后將當(dāng)前對(duì)象的非靜態(tài)字段復(fù)制到該新對(duì)象,無論該字段是值類型的還是引用類型,都復(fù)制獨(dú)立的一份。當(dāng)你修改其中一個(gè)對(duì)象的任何內(nèi)容時(shí),都不會(huì)影響另一個(gè)對(duì)象的內(nèi)容。圖片 那么該如何實(shí)現(xiàn)深拷貝呢?Object 類提供的 clone 是只能實(shí)現(xiàn) 淺拷貝的。

6、如何實(shí)現(xiàn)深拷貝?

深拷貝的原理我們知道了,就是要讓原始對(duì)象和克隆之后的對(duì)象所具有的引用類型屬性不是指向同一塊堆內(nèi)存,這里有兩種實(shí)現(xiàn)思路。

①、讓每個(gè)引用類型屬性內(nèi)部都重寫clone() 方法

既然引用類型不能實(shí)現(xiàn)深拷貝,那么我們將每個(gè)引用類型都拆分為基本類型,分別進(jìn)行淺拷貝。比如上面的例子,Person 類有一個(gè)引用類型 Address(其實(shí)String 也是引用類型,但是String類型有點(diǎn)特殊,后面會(huì)詳細(xì)講解),我們?cè)?Address 類內(nèi)部也重寫 clone 方法。如下:

Address.class:

package com.ys.test;

public class Address implements Cloneable{
    private String provices;
    private String city;
    public void setAddress(String provices,String city){
        this.provices = provices;
        this.city = city;
    }
    @Override
    public String toString() {
        return "Address [provices=" + provices + ", city=" + city + "]";
    }
    @Override
    protected Object clone() throws CloneNotSupportedException {
        return super.clone();
    }

}

Person.class 的 clone() 方法:

@Override
    protected Object clone() throws CloneNotSupportedException {
        Person p = (Person) super.clone();
        p.address = (Address) address.clone();
        return p;
    }

測試還是和上面一樣,我們會(huì)發(fā)現(xiàn)更改了p2對(duì)象的Address屬性,p1 對(duì)象的 Address 屬性并沒有變化。

但是這種做法有個(gè)弊端,這里我們Person 類只有一個(gè) Address 引用類型,而 Address 類沒有,所以我們只用重寫 Address 類的clone 方法,但是如果 Address 類也存在一個(gè)引用類型,那么我們也要重寫其clone 方法,這樣下去,有多少個(gè)引用類型,我們就要重寫多少次,如果存在很多引用類型,那么代碼量顯然會(huì)很大,所以這種方法不太合適。

②、利用序列化

序列化是將對(duì)象寫到流中便于傳輸,而反序列化則是把對(duì)象從流中讀取出來。這里寫到流中的對(duì)象則是原始對(duì)象的一個(gè)拷貝,因?yàn)樵紝?duì)象還存在 JVM 中,所以我們可以利用對(duì)象的序列化產(chǎn)生克隆對(duì)象,然后通過反序列化獲取這個(gè)對(duì)象。

注意每個(gè)需要序列化的類都要實(shí)現(xiàn) Serializable 接口,如果有某個(gè)屬性不需要序列化,可以將其聲明為 transient,即將其排除在克隆屬性之外。

//深度拷貝
public Object deepClone() throws Exception{
    // 序列化
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    ObjectOutputStream oos = new ObjectOutputStream(bos);

    oos.writeObject(this);

    // 反序列化
    ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
    ObjectInputStream ois = new ObjectInputStream(bis);

    return ois.readObject();
}

因?yàn)樾蛄谢a(chǎn)生的是兩個(gè)完全獨(dú)立的對(duì)象,所有無論嵌套多少個(gè)引用類型,序列化都是能實(shí)現(xiàn)深拷貝的。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 數(shù)據(jù)
    +關(guān)注

    關(guān)注

    8

    文章

    7250

    瀏覽量

    91504
  • JAVA
    +關(guān)注

    關(guān)注

    20

    文章

    2987

    瀏覽量

    108173
  • 源碼
    +關(guān)注

    關(guān)注

    8

    文章

    669

    瀏覽量

    30240
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    #硬聲創(chuàng)作季 4-9 Mat的拷貝拷貝

    數(shù)據(jù)庫
    Mr_haohao
    發(fā)布于 :2022年08月30日 15:32:51

    C++零基礎(chǔ)教程之C++拷貝拷貝,輕松上手C++拷貝拷貝

    編程語言C++語言
    電子學(xué)習(xí)
    發(fā)布于 :2023年01月14日 11:37:32

    32.[程序員小飛]用最淺顯的語言,深入理解拷貝拷貝的區(qū)別

    程序代碼
    充八萬
    發(fā)布于 :2023年07月20日 02:25:58

    拷貝拷貝的實(shí)現(xiàn)方法概述

    拷貝拷貝的實(shí)現(xiàn)
    發(fā)表于 07-19 13:35

    python深淺拷貝是什么?

    python的直接賦值python的拷貝python的拷貝
    發(fā)表于 11-04 08:33

    請(qǐng)問哪位大神可以詳細(xì)介紹JavaScript拷貝拷貝

    JavaScript數(shù)據(jù)類型JavaScript拷貝拷貝
    發(fā)表于 11-05 07:16

    光盤拷貝機(jī)的使用端口

    光盤拷貝機(jī)的使用端口              使用端口指的是:光盤拷貝機(jī)內(nèi)部控制器與刻錄機(jī)相連的接口。目前光盤拷貝
    發(fā)表于 12-30 10:04 ?1061次閱讀

    C#拷貝拷貝區(qū)別解析

     所謂拷貝就是將對(duì)象中的所有字段復(fù)制到新的副本對(duì)象中;拷貝對(duì)于值類型與引用類型的方式有區(qū)別,值類型字段的值被復(fù)制到副本中后,在副本中的修改不會(huì)影響源對(duì)象對(duì)應(yīng)的值;然而對(duì)于引用類型的
    發(fā)表于 11-29 08:32 ?2.6w次閱讀
    C#<b class='flag-5'>淺</b><b class='flag-5'>拷貝</b>與<b class='flag-5'>深</b><b class='flag-5'>拷貝</b>區(qū)別解析

    Python如何防止數(shù)據(jù)被修改Python中的拷貝拷貝的問題說明

    在平時(shí)工作中,經(jīng)常涉及到數(shù)據(jù)的傳遞。在數(shù)據(jù)傳遞使用過程中,可能會(huì)發(fā)生數(shù)據(jù)被修改的問題。為了防止數(shù)據(jù)被修改,就需要再傳遞一個(gè)副本,即使副本被修改,也不會(huì)影響原數(shù)據(jù)的使用。為了生成這個(gè)副本,就產(chǎn)生了拷貝——今天就說一下Python中的拷貝
    的頭像 發(fā)表于 03-30 09:54 ?3291次閱讀
    Python如何防止數(shù)據(jù)被修改Python中的<b class='flag-5'>深</b><b class='flag-5'>拷貝</b>與<b class='flag-5'>淺</b><b class='flag-5'>拷貝</b>的問題說明

    C++之拷貝構(gòu)造函數(shù)的copy及copy

    C++編譯器會(huì)默認(rèn)提供構(gòu)造函數(shù);無參構(gòu)造函數(shù)用于定義對(duì)象的默認(rèn)初始化狀態(tài);拷貝構(gòu)造函數(shù)在創(chuàng)建對(duì)象時(shí)拷貝對(duì)象的狀態(tài);對(duì)象的拷貝拷貝
    的頭像 發(fā)表于 12-24 15:31 ?975次閱讀

    為什么有時(shí)候會(huì)寫出爛代碼

    就寫出來了呢?其實(shí)還是因?yàn)橛行┲R(shí)沒那么扎實(shí)了~就容易被忽略了,于是我在團(tuán)隊(duì)群里面強(qiáng)調(diào)了一下這個(gè)問題: 所以,本文主要是關(guān)于BeanUtils工具的屬性拷貝以及拷貝
    的頭像 發(fā)表于 08-27 10:23 ?1563次閱讀
    為什么有時(shí)候會(huì)寫出爛代碼

    C++面向?qū)ο缶幊讨械?b class='flag-5'>深拷貝拷貝

    可能對(duì)于Java程序員來說,很少遇到深淺拷貝問題,但是對(duì)于C++程序員來說可謂是又愛又恨。。
    的頭像 發(fā)表于 03-30 12:53 ?1071次閱讀
    C++面向?qū)ο缶幊讨械?b class='flag-5'>深</b><b class='flag-5'>拷貝</b>和<b class='flag-5'>淺</b><b class='flag-5'>拷貝</b>

    C++拷貝拷貝詳解

    當(dāng)類的函數(shù)成員存在指針成員時(shí)會(huì)產(chǎn)生拷貝拷貝和問題。
    發(fā)表于 08-21 15:05 ?555次閱讀
    C++<b class='flag-5'>深</b><b class='flag-5'>拷貝</b>和<b class='flag-5'>淺</b><b class='flag-5'>拷貝</b>詳解

    Python中拷貝拷貝的操作

    【例子】拷貝拷貝中 list1 = [ 123 , 456 , 789 , 213 ]list2 = list1list3 = list1[:] print (list2) #
    的頭像 發(fā)表于 11-02 10:58 ?599次閱讀

    磁盤拷貝機(jī)會(huì)拷貝刪除的內(nèi)容嗎

    磁盤拷貝機(jī),也稱為硬盤克隆器或磁盤復(fù)制器,是一種用于復(fù)制硬盤驅(qū)動(dòng)器內(nèi)容的設(shè)備。它可以將一個(gè)硬盤上的所有數(shù)據(jù),包括操作系統(tǒng)、程序、文件和設(shè)置,復(fù)制到另一個(gè)硬盤上。這種設(shè)備在數(shù)據(jù)備份、系統(tǒng)遷移、硬盤
    的頭像 發(fā)表于 10-14 15:38 ?1203次閱讀
    主站蜘蛛池模板: 色多多在线免费观看 | 国产精品主播在线 | 成人免费无毒在线观看网站 | 三级精品视频在线播放 | 操www| 天堂在线.www资源在线观看 | 欧美女同在线观看 | 手机看日韩毛片福利盒子 | 亚洲视屏一区 | 天天色综合社区 | 亚洲国内精品自在线影视 | 美日韩中文字幕 | 黄色网页在线播放 | 国产精品视频网站你懂得 | 亚洲国产成人久久三区 | 六月婷婷激情综合 | 青青草国产三级精品三级 | 日本写真高清视频免费网站网 | 91日本在线观看亚洲精品 | 在线免费看高清视频大全 | 日韩高清在线日韩大片观看网址 | 亚洲精品成人a在线观看 | 最刺激黄a大片免费观看下截 | 国产成人毛片亚洲精品不卡 | 国产精品美女www爽爽爽视频 | 色综合久久98天天综合 | 亚洲大色 | 大香线蕉97久久 | 午色影院| www.午夜视频 | 男女交性无遮挡免费视频 | www.淫.com| 欧美性色黄大片四虎影视 | 精品国产免费观看一区高清 | 成人一级视频 | 六月综合 | 美国一级毛片免费看成人 | 狠狠色噜噜狠狠狠狠2021天天 | 最好看最新的中文字幕1 | 亚洲欧洲精品成人久久曰影片 | 99久久精品免费观看国产 |