国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

簡單聊聊對象淺拷貝和深拷貝,真不簡單!

開發 前端
本文主要圍繞對象的淺拷貝和深拷貝,從使用方面做了一次簡單的內容總結。淺拷貝下,原對象和目標對象,引用都是同一個對象,當被引用的對象數據發生變化時,相關的引用者也會跟著一起變。

一、摘要

上篇文章中,我們有介紹到對象屬性復制相關的工具,這些工具所進行的對象拷貝,其實都是淺拷貝模式。

可能有的同學會發出疑問,什么叫淺拷貝?

我們都知道,Java 中的數據類型分為值類型(基本數據類型)和引用類型,值類型包括 byte、short、 int、long、float、double、boolean、char 等簡單數據類型,引用類型包括類、接口、數組等復雜類型。

根據數據類型的不同,在進行屬性值拷貝的時候,如果是值類型,復制的是屬性值,如果是復雜類型,比如對象,復制的內容可能是屬性對應的內存引用地址。

因此,在 Java 中對于復雜類型的數據,也分為**淺拷貝(淺克隆)與深拷貝(深克隆)**方式,區別如下:

  • 淺拷貝:將原對象或原數組的引用直接賦給新對象或者新數組,新對象只是原對象的一個引用,也就是說不管新對象還是原對象,都是引用同一個對象
  • 深拷貝:創建一個新的對象或者數組,將原對象的各項屬性的值拷貝過來,是“值”而不是“引用”,兩者對象是不一樣的

對于概念的解釋,可能也很難理解,下面我們簡單的通過幾個案例向大家介紹!

二、案例實踐

2.1、淺拷貝

首先我們新建兩個對象,其中User關聯Account對象,內容如下:

public class User {

    /**
     * 用戶ID
     */
    private Long userId;

    /**
     * 賬戶信息
     */
    private Account account;

    //...get、set

    @Override
    public String toString() {
        return "User{" +
                "userId=" + userId +
                ", account=" + account +
                '}';
    }
}
public class Account {

    /**
     * 賬號余額
     */
    private BigDecimal money;

    //...get、set

    @Override
    public String toString() {
        return "Account{" +
                "money=" + money +
                '}';
    }
}

使用Spring BeanUtils工具進行對象屬性復制,操作如下:

// 定義某用戶,賬戶余額 100塊
Account sourceAccount = new Account();
sourceAccount.setMoney(BigDecimal.valueOf(100));

User sourceUser = new User();
sourceUser.setUserId(1L);
sourceUser.setAccount(sourceAccount);

// 進行對象屬性拷貝
User targetUser = new User();
BeanUtils.copyProperties(sourceUser, targetUser);
System.out.println("修改嵌套對象屬性值前的結果:" + targetUser.toString());

//修改原始對象賬戶余額為200
sourceAccount.setMoney(BigDecimal.valueOf(200));

System.out.println("修改嵌套對象屬性值后的結果:" + targetUser.toString());

輸出結果如下:

修改嵌套對象屬性值前的結果:User{userId=1, account=Account{money=100}}
修改嵌套對象屬性值后的結果:User{userId=1, account=Account{money=200}}

從結果上可以很明顯的得出結論:當修改原始的嵌套對象Account的屬性值時,目標對象的Account對象對應的值也跟著發生變化。

很顯然,這與我們預期想要的對象屬性拷貝是想違背的,我們所期待的結果是:原始對象值即使發生變化,目標對象的值也不應該發生變化!

面對這種情況,怎么處理呢?

我們可以把對象Account單獨拉出來,進行一次屬性值拷貝,然后再進行封裝,比如操作如下:

// 定義某用戶,賬戶余額 100塊
Account sourceAccount = new Account();
sourceAccount.setMoney(BigDecimal.valueOf(100));

User sourceUser = new User();
sourceUser.setUserId(1L);
sourceUser.setAccount(sourceAccount);


// 拷貝 Account 對象
Account targetAccount = new Account();
BeanUtils.copyProperties(sourceAccount, targetAccount);

// 拷貝 User 對象
User targetUser = new User();
BeanUtils.copyProperties(sourceUser, targetUser);
targetUser.setAccount(targetAccount);
System.out.println("修改嵌套對象屬性值前的結果:" + targetUser.toString());

//修改原始對象賬戶余額為200
sourceAccount.setMoney(BigDecimal.valueOf(200));

System.out.println("修改嵌套對象屬性值后的結果:" + targetUser.toString());

輸出結果如下:

修改嵌套對象屬性值前的結果:User{userId=1, account=Account{money=100}}
修改嵌套對象屬性值后的結果:User{userId=1, account=Account{money=100}}

即使Account對象數據發生變化,也不會改目標對象的數據,與預期結果一致!

現在的情況是User只有一個嵌套對象Account,假如像這樣的對象有十幾個呢,采用以上方式顯然不可取。

這個時候深拷貝,該登場了!

2.2、深拷貝

Java 的深拷貝有兩種實現方式,第一種是通過將對象序列化到臨時文件,然后再通過反序列化方式,從臨時文件中讀取數據,操作案例如下!

首先所有的類,必須實現Serializable接口,推薦顯式定義序列化 ID。

public class User implements Serializable {

    private static final long serialVersionUID = 1L;

    /**
     * 用戶ID
     */
    private Long userId;

    /**
     * 賬戶信息
     */
    private Account account;

    //...get、set

    @Override
    public String toString() {
        return "User{" +
                "userId=" + userId +
                ", account=" + account +
                '}';
    }
}
public class Account implements Serializable {

    private static final long serialVersionUID = 1L;

    /**
     * 賬號余額
     */
    private BigDecimal money;

    //...get、set

    @Override
    public String toString() {
        return "Account{" +
                "money=" + money +
                '}';
    }
}
// 定義某用戶,賬戶余額 100塊
Account sourceAccount = new Account();
sourceAccount.setMoney(BigDecimal.valueOf(100));

User sourceUser = new User();
sourceUser.setUserId(1L);
sourceUser.setAccount(sourceAccount);


//把對象寫入文件中
try {
    FileOutputStream fos = new FileOutputStream("temp.out");
    ObjectOutputStream oos = new ObjectOutputStream(fos);
    oos.writeObject(sourceUser);
    oos.flush();
    oos.close();
} catch (IOException e) {
    e.printStackTrace();
}

//從文件中讀取對象
User targetUser = null;
try {
    FileInputStream fis = new FileInputStream("temp.out");
    ObjectInputStream ois = new ObjectInputStream(fis);
    targetUser = (User) ois.readObject();
    fis.close();
    ois.close();
}  catch (Exception e) {
    e.printStackTrace();
}

System.out.println("修改嵌套對象屬性值前的結果:" + targetUser.toString());

//修改原始對象賬戶余額為200
sourceAccount.setMoney(BigDecimal.valueOf(200));

System.out.println("修改嵌套對象屬性值后的結果:" + targetUser.toString());

輸出結果:

修改嵌套對象屬性值前的結果:User{userId=1, account=Account{money=100}}
修改嵌套對象屬性值后的結果:User{userId=1, account=Account{money=100}}

通過序列化和反序列化的方式,可以實現多層復雜的對象數據拷貝。

因為涉及到需要將數據寫入臨時磁盤,性能可能會有所下降!

2.3、json 序列化和反序列化

對于對象深度拷貝,還有第二種方式,那就是采用 json 序列化和反序列化相關的技術來實現,同時性能也比將數據寫入臨時磁盤的方式要好很多,并且不需要顯式實現序列化接口。

json 序列化和反序列化的底層思想是,將對象序列化成字符串;然后再將字符串通過反序列化方式成對象。

以jackson工具庫為例,具體使用方式如下!

首先導入相關的jackson依賴包!

<!--jackson依賴-->
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-core</artifactId>
    <version>2.9.8</version>
</dependency>
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-annotations</artifactId>
    <version>2.9.8</version>
</dependency>
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.9.8</version>
</dependency>

其次,編寫統一Json處理工具類!

public class JsonUtil {

    private static final Logger log = LoggerFactory.getLogger(JsonUtil.class);

    private static ObjectMapper objectMapper = new ObjectMapper();

    static {
        // 序列化時,將對象的所有字段全部列入
        objectMapper.setSerializationInclusion(JsonInclude.Include.ALWAYS);
        // 允許沒有引號的字段名
        objectMapper.configure(JsonParser.Feature.ALLOW_UNQUOTED_FIELD_NAMES, true);
        // 自動給字段名加上引號
        objectMapper.configure(JsonGenerator.Feature.QUOTE_FIELD_NAMES, true);
        // 時間默認以時間戳格式寫,默認時間戳
        objectMapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, true);
        // 忽略空bean轉json的錯誤
        objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
        // 設置時間轉換所使用的默認時區
        objectMapper.setTimeZone(TimeZone.getDefault());


        // 反序列化時,忽略在json字符串中存在, 但在java對象中不存在對應屬性的情況, 防止錯誤
        objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
        objectMapper.configure(MapperFeature.ACCEPT_CASE_INSENSITIVE_PROPERTIES, true);

        //序列化/反序列化,自定義設置
        SimpleModule module = new SimpleModule();
        // 序列化成json時,將所有的long變成string
        module.addSerializer(Long.class, ToStringSerializer.instance);
        module.addSerializer(Long.TYPE, ToStringSerializer.instance);
        // 自定義參數配置注冊
        objectMapper.registerModule(module);
    }

    /**
     * 對象序列化成字符串
     * @param obj
     * @param <T>
     * @return
     */
    public static <T> String objToStr(T obj) {
        if (null == obj) {
            return null;
        }

        try {
            return obj instanceof String ? (String) obj : objectMapper.writeValueAsString(obj);
        } catch (Exception e) {
            log.warn("objToStr error: ", e);
            return null;
        }
    }

    /**
     * 字符串反序列化成對象
     * @param str
     * @param clazz
     * @param <T>
     * @return
     */
    public static <T> T strToObj(String str, Class<T> clazz) {
        try {
            return clazz.equals(String.class) ? (T) str : objectMapper.readValue(str, clazz);
        } catch (Exception e) {
            log.warn("strToObj error: ", e);
            return null;
        }
    }

    /**
     * 字符串反序列化成對象(數組)
     * @param str
     * @param typeReference
     * @param <T>
     * @return
     */
    public static <T> T strToObj(String str, TypeReference<T> typeReference) {
        try {
            return (T) (typeReference.getType().equals(String.class) ? str : objectMapper.readValue(str, typeReference));
        } catch (Exception e) {
            log.warn("strToObj error", e);
            return null;
        }
    }
}

最后,在相關的位置引入即可。

// 定義某用戶,賬戶余額 100塊
Account sourceAccount = new Account();
sourceAccount.setMoney(BigDecimal.valueOf(100));

User sourceUser = new User();
sourceUser.setUserId(1L);
sourceUser.setAccount(sourceAccount);

// json序列化、反序列化
User targetUser = JsonUtil.strToObj(JsonUtil.objToStr(sourceUser), User.class);
System.out.println("修改嵌套對象屬性值前的結果:" + targetUser.toString());

//修改原始對象賬戶余額為200
sourceAccount.setMoney(BigDecimal.valueOf(200));

System.out.println("修改嵌套對象屬性值后的結果:" + targetUser.toString());

輸出結果:

修改嵌套對象屬性值前的結果:User{userId=1, account=Account{money=100}}
修改嵌套對象屬性值后的結果:User{userId=1, account=Account{money=100}}

與預期一致!

三、小結

本文主要圍繞對象的淺拷貝和深拷貝,從使用方面做了一次簡單的內容總結。

淺拷貝下,原對象和目標對象,引用都是同一個對象,當被引用的對象數據發生變化時,相關的引用者也會跟著一起變。

深拷貝下,原對象和目標對象數據是兩個完全獨立的存在,相互直接不受影響。

至于當前對象數據,是應該走淺拷貝還是深拷貝模式好,完全取決于當前業務的需求,沒有絕對的好或者不好!

如果當前對象需要深拷貝,推薦采用 json 序列化和反序列化的方式實現,相比通過文件寫入的方式進行序列化和反序列化,操作簡單且性能高!

責任編輯:武曉燕 來源: Java極客技術
相關推薦

2021-10-19 08:20:47

單例模式設計模式面試

2014-02-24 14:45:23

XPath開發工具

2020-12-16 07:36:46

Redis字符串數據

2017-12-25 15:35:36

iMac Pro芯片存儲

2017-08-16 13:30:05

Java深拷貝淺拷貝

2021-07-16 12:33:24

Javascript深拷貝淺拷貝

2022-07-26 08:07:03

Python淺拷貝深拷貝

2021-09-27 11:07:11

深拷貝淺拷貝內存

2020-08-03 08:24:26

原型模式拷貝

2018-09-26 14:37:17

JavaScript前端編程語言

2021-01-08 06:15:09

深拷貝淺拷貝寫時拷貝

2023-05-17 08:42:46

深拷貝Golang

2009-05-19 17:28:44

深拷貝淺拷貝clone()

2023-05-05 08:47:35

Java淺拷貝深拷貝

2020-10-12 08:35:22

JavaScript

2024-03-15 15:03:23

2025-04-27 09:45:58

JavaScript深拷貝淺拷貝

2024-08-02 08:43:24

JavaScript開發者工具箱深拷貝

2018-05-10 14:20:18

前端JavaScript深拷貝

2022-09-30 15:03:09

C語言深拷貝淺拷貝
點贊
收藏

51CTO技術棧公眾號

四虎成人在线视频| 女人丝袜激情亚洲| 超碰人人在线| 秋霞综合在线视频| 国产一区二区不卡视频| 黄页视频在线免费观看| 国产精品麻豆久久久| 91xxxxx| 色8久久精品久久久久久蜜| 在线的色视频| 天天亚洲美女在线视频| 狠狠做六月爱婷婷综合aⅴ| 国内少妇毛片视频| 亚洲成人黄色在线观看| 秋霞蜜臀av久久电影网免费| 韩国一区二区三区美女美女秀| 狠狠色丁香久久综合频道| 国产乱人伦真实精品视频| 98精品久久久久久久| 一级黄色录像免费看| 日韩精品一区二区三区视频播放| 国产主播性色av福利精品一区| 97se国产在线视频| 综合电影一区二区三区 | 欧美视频你懂的| 日本韩国在线视频爽| 欧美一区二区免费观在线| 天堂蜜桃一区二区三区| 男人av在线播放| 一区二区三区一级片| 色狠狠一区二区| 看全色黄大色大片免费久久久| 亚洲国产精品毛片| 亚洲免费高清视频在线| 青青草原av在线| 国产女教师bbwbbwbbw| 91精品国产综合久久蜜臀| 日韩欧美黄色| www.浪潮av.com| 色噜噜夜夜夜综合网| 久久久www| 欧美久久电影| 欧美在线观看一区二区| 精品国产网站| 日韩视频在线视频| 在线播放/欧美激情| 成人免费在线播放视频| 精品久久久久久最新网址| 大桥未久av一区二区三区中文| 亚洲免费二区| 男女视频在线| 国产日韩亚洲欧美在线| 久久久国产一区二区| 日本在线观看大片免费视频| 久久精品国产精品亚洲精品色| 91中文精品字幕在线视频| 久久久国产成人精品| 性日韩欧美在线视频| 日韩麻豆第一页| 精品久久久久久久久久久久久久久| 在线观看91视频| 成人国产免费视频| 成人sese在线| 国产传媒一区在线| 国产成人精品亚洲777人妖 | 亚洲成在线观看| 972aa.com艺术欧美| 一区二区三区小说| 亚洲精品理论电影| 欧美日韩成人网| 九九九久久久| 伊人国产在线视频| av在线最新| 国产影视一区| av一区二区三区黑人| 亚洲一区二区欧美日韩| 色综合中文字幕| 亚洲美女性视频| 成人国产精品久久久久久亚洲| 国产综合在线观看视频| 国产91色在线|| 日韩精品在线观看av| 在线播放日本| 欧美艳星介绍134位艳星| 91日韩一区二区三区| 精品视频在线导航| 97久久夜色精品国产九色| 少妇人妻在线视频| 超碰97免费在线| 欧美电影免费网站| 久久电影国产免费久久电影 | 欧美成人一品| 久久综合色综合88| 欧美狂野另类xxxxoooo| 久久手机精品视频| 国产乱人伦精品一区二区| 日本wwwcom| 日本中文字幕在线播放| 日韩精品网站| 国产精品污网站| 亚洲精选在线观看| 国产亚洲一区二区三区在线播放| 欧美日韩国产高清视频| 制服黑丝国产在线| 色悠久久久久综合先锋影音下载| 精品久久久久久久久国产字幕| 久久久久亚洲蜜桃| 黄色成人在线网| 在线āv视频| 国产精品久久久久久久久久白浆| 91精品综合| 亚洲专区一区二区三区| 国产成人在线看| 午夜av一区二区| 国产一区二区动漫| 国产欧美精品一区二区三区介绍 | 超碰成人av| 亚洲色图美女| 久久激情婷婷| 日韩高清免费观看| 亚洲色图清纯唯美| 欧美性一级生活| 亚洲国产精品字幕| 久久91超碰青草是什么| 91久久久亚洲精品| 亚洲熟妇av一区二区三区| 91福利在线视频| 丁香综合av| 亚洲国产精品91| 麻豆精品视频在线| 中文字幕一区二区三区精华液| 精品福利在线看| 色综合久久88色综合天天 | 欧美激情精品久久久久久久变态| 久久精品国产欧美激情| 亚洲欧美日韩国产精品| 日韩一级片网站| 日韩电影中文 亚洲精品乱码| 精品国产麻豆免费人成网站| 91高清视频在线免费观看| 久久久无码中文字幕久...| 青草久久伊人| 色综合中文网| 国产一区视频导航| 欧美日韩美女一区二区| 欧美在线视频在线播放完整版免费观看 | 国产精品91一区二区三区| 亚洲白虎美女被爆操| 成人激情电影一区二区| 日韩精品极品视频| 69av成年福利视频| 国精产品一区一区三区视频| 中文字幕日本最新乱码视频| 最近中文字幕mv免费高清在线| 色呦呦在线观看视频| 99久久影视| 一区二区激情小说| 婷婷开心激情综合| 亚洲国产另类久久精品| 欧美成人性生活| 2019av中文字幕| 韩国日本不卡在线| 久久久久久久久网| 在线观看av影片| 国产极品一区| 成人h动漫精品一区二| 欧美精品一区二区三区蜜桃视频 | 日本中文不卡| av一区二区在线观看| 91视频一区| 日本精品一级二级| 成人永久免费| 欧美成人二区| 国产成人精品三级麻豆| 国产成人精品综合| 小视频免费在线观看| 亚洲影院在线观看| av日韩在线看| 一区二区国产在线观看| 久久久精品免费观看| 日韩中文在线中文网在线观看 | 日本高清视频一区| 精品国产乱码| 亚洲伦理中文字幕| 中文字幕网在线| 99久久精品国产观看| 国产原创精品| 波多野结衣在线播放一区| 亚洲一区二区福利| 粉嫩91精品久久久久久久99蜜桃| 日韩精品一区二区在线| 国产精品www在线观看| 亚洲老妇激情| 久久久精品久久久| 久久77777| 欧美性感美女h网站在线观看免费| 老熟妇仑乱视频一区二区| 国产福利一区在线| 无码人妻精品一区二区蜜桃网站| 亚洲日本中文字幕区|