創建型模式之原型模式(2.3)

来源:https://www.cnblogs.com/noneplus/archive/2019/08/12/11342048.html
-Advertisement-
Play Games

簡單來說,通過複製的方式創建對象。 【舉個慄子】:點外賣的收貨地址 ...


什麼是原型模式?

簡單來說,通過複製的方式創建對象。(被覆制的對象可以理解為模板)


原型模式的應用場景

複雜結構對象的創建。

複雜結構對象:可以理解為對象裡面還有對象。


【舉個慄子】:點外賣的收貨地址


收貨地址包括姓名,電話和住址。第一次點外賣的時候需要完整的填寫這些信息,但是之後點的過程基本上是不用再重新填寫這些信息的。試想一下,每次下單之前都要填一遍姓名,電話,詳細地址(假設省市已經定位好了)。頭大.....而這個場景,正是原型模式的用武之地。

點外賣之前複製上一次填寫的地址,直接下單;或者手機號出現變更,修改一下直接更新到模板即可。

原型模式的本質是創建一個對象模板,然後通過複製的方式實現復用。


深拷貝(複雜對象一波帶走)

在深拷貝之前,簡單聊一下淺拷貝。

以上述收貨地址為例,如果只涉及到姓名,電話這些基本數據類型(淺拷貝與深拷貝是基本沒有區別的),但是如果涉及到地址這樣的引用類型。淺拷貝就會出現問題。淺拷貝對於引用對象,只複製其引用地址。所以複製出來的對象和被覆制的對象會指向同一個值。假設你想建立兩個收貨模板,地址是不同的,那麼就只能使用深拷貝了。


擼段代碼試試看:

思路:

1.創建Info類和Address類繼承Serializable介面,為了後面實現序列化寫入記憶體。(實現平臺無關性)

2.Info類還需要繼承Cloneable介面,並重寫clone()方法,註:protected改為public

3.在clone()方法中添加寫入記憶體和讀取的邏輯

4.Info info1 = (Info) info.clone();創建對象,並可做靈活修改。

import java.io.*;

class Address implements  Serializable
{
    private String province;
    private String city;
    private String Street;
    private String door_number;

    public Address(String province, String city, String street, String door_number) {
        this.province = province;
        this.city = city;
        Street = street;
        this.door_number = door_number;
    }

    public String getProvince() {
        return province;
    }

    public void setProvince(String province) {
        this.province = province;
    }

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

    public String getStreet() {
        return Street;
    }

    public void setStreet(String street) {
        Street = street;
    }

    public String getDoor_number() {
        return door_number;
    }

    public void setDoor_number(String door_number) {
        this.door_number = door_number;
    }

    @Override
    public String toString() {
        return "Address{" +
                "province='" + province + '\'' +
                ", city='" + city + '\'' +
                ", Street='" + Street + '\'' +
                ", door_number='" + door_number + '\'' +
                '}';
    }
}

class Info implements Cloneable, Serializable   //1.創建一個外賣信息類,繼承Cloneable(可複製),Serializable(可序列化)
{
    private String name;
    private String number;

    Address address;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getNumber() {
        return number;
    }

    public void setNumber(String number) {
        this.number = number;
    }

    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

    public Info(String name, String number, Address address) {
        this.name = name;
        this.number = number;
        this.address = address;
    }

    @Override
    public String toString() {
        return "Info{" +
                "name='" + name + '\'' +
                ", number='" + number + '\'' +
                ", address=" + address +
                '}';
    }

    @Override
    public Object clone() throws CloneNotSupportedException {    //2.重寫clone方法,並把protected換成public
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        try {
                ObjectOutputStream oos = new ObjectOutputStream(out);
                oos.writeObject(this);   //寫入記憶體
                oos.close();

            byte[] bytes = out.toByteArray();
            InputStream in = new ByteArrayInputStream(bytes);
            ObjectInputStream ois = new ObjectInputStream(in);
            Object clone = ois.readObject();  //讀取記憶體
            ois.close();

            return clone;

        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

}

/*
clone()方法直接複製在記憶體中已經創建對象的二進位,效率極高!(不需要調用構造器)

protected native Object clone() throws CloneNotSupportedException;
native修飾的方法直接調用底層的C語言

 */
public class MyInfo
{
    public static void main(String[] args) throws CloneNotSupportedException {
        Info info = new Info("shadow","18116207310",new Address("上海","浦東新區","振南路","355"));
        System.out.println(info);
        Info info1 = (Info) info.clone();  //3.複製加強轉
        System.out.println(info1);

        System.out.println("//修改電話:");
        Info info2 = (Info) info.clone();
        info2.setNumber("123456789");
        System.out.println(info2);

        System.out.println("//修改地址:");
        Info info3 = (Info) info.clone();
        info3.getAddress().setStreet("西語街");
        info3.getAddress().setDoor_number("666");

        System.out.println(info3);
    }
}

輸出結果:

1565608154656


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 製作一個字元串的跑馬燈效果 (1)實例代碼 (2)摘要 實現案例的基本思路是利用substring截取字元串再進行拼接,然後使用定時器來達到動態的效果。 箭頭函數可以解決this的指向問題,箭頭函數內部的this相對於外部this的指向。 需要訪問data中數據時,一定要使用this訪問,比如thi ...
  • 寫在前面:Javascript 中Math和其他對象不同,它具有數學常數和函數的屬性和方法。因為它的屬性是數學常數,所以不能被改變(可以進行賦值操作,但最後值不變)。 Math的方法就是普通函數,調用他們直接用Math.method即可,而無需使用new關鍵字創造它的實例。JS中的其他對象也有類似這 ...
  • 1遞歸含義:在某時某刻某個條件下調用包含自己的函數 2:註意點:⑴遞歸過程中一定要加限制條件,要不然會陷入死迴圈: 死迴圈eg: 正常調用: ⑵遞歸有個過程,不是一步到位的,這一點尤其重要,因為在學習JS數據結構與演算法中的二叉搜索樹的移除代碼會至關重要,不懂遞歸過程的話很容易看不懂移除代碼 過程如下 ...
  • 一...........三種引入.................. 1.內部引入 <script type="text/javascript"> </script> 2.行內引入 <a href="javascript:confirm('你確定要報名嗎');" 報名</a> <p onclick= ...
  • 1.開關燈效果 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <style> .hide{ display: none; } </style> </head> <body> <i ...
  • vue 父子組件傳值是很常見的,多數情況下都是父傳遞給子的值是基礎數據類型,如string,number,boolean, 當父組件值被修改時,子組件能夠實時的作出改變。 如果父子傳值的類型是複雜數據類型(object,array)這種時, 1.通常的做法是在子組件監聽props屬性 細節點:這裡監 ...
  • 什麼是建造者模式? 工廠模式聚焦於創建出一個對象,而建造者除此之外還需要為創建的對象賦值。 簡單來說,建造者模式=創建對象+屬性賦值。 建造者模式應用場景 建造者模式適合創建 類中包含多個參數且需要定製化 的情況。 簡單來說,建造者模式的目的就是創造 一條龍服務 :不僅創建出對象,順便給屬性賦值。 ...
  • 前言 我們已經學習了單一職責原則,依賴倒置原則,介面隔離原則,李氏替換原則。可以說前面幾個原則都是為了開閉原則奠定基礎。 我們寫的程式由於實際的情況可以一定程度上違背各種設計原則。但是,開閉原則我認為作為一個程式猿無論什麼時候都需要遵循他,切記不可違背她。 基本介紹 1. 開閉原則(Open Clo ...
一周排行
    -Advertisement-
    Play Games
  • 概述:本文代碼示例演示瞭如何在WPF中使用LiveCharts庫創建動態條形圖。通過創建數據模型、ViewModel和在XAML中使用`CartesianChart`控制項,你可以輕鬆實現圖表的數據綁定和動態更新。我將通過清晰的步驟指南包括詳細的中文註釋,幫助你快速理解並應用這一功能。 先上效果: 在 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • openGauss(GaussDB ) openGauss是一款全面友好開放,攜手伙伴共同打造的企業級開源關係型資料庫。openGauss採用木蘭寬鬆許可證v2發行,提供面向多核架構的極致性能、全鏈路的業務、數據安全、基於AI的調優和高效運維的能力。openGauss深度融合華為在資料庫領域多年的研 ...
  • 概述:本示例演示了在WPF應用程式中實現多語言支持的詳細步驟。通過資源字典和數據綁定,以及使用語言管理器類,應用程式能夠在運行時動態切換語言。這種方法使得多語言支持更加靈活,便於維護,同時提供清晰的代碼結構。 在WPF中實現多語言的一種常見方法是使用資源字典和數據綁定。以下是一個詳細的步驟和示例源代 ...
  • 描述(做一個簡單的記錄): 事件(event)的本質是一個委托;(聲明一個事件: public event TestDelegate eventTest;) 委托(delegate)可以理解為一個符合某種簽名的方法類型;比如:TestDelegate委托的返回數據類型為string,參數為 int和 ...
  • 1、AOT適合場景 Aot適合工具類型的項目使用,優點禁止反編 ,第一次啟動快,業務型項目或者反射多的項目不適合用AOT AOT更新記錄: 實實在在經過實踐的AOT ORM 5.1.4.117 +支持AOT 5.1.4.123 +支持CodeFirst和非同步方法 5.1.4.129-preview1 ...
  • 總說周知,UWP 是運行在沙盒裡面的,所有許可權都有嚴格限制,和沙盒外交互也需要特殊的通道,所以從根本杜絕了 UWP 毒瘤的存在。但是實際上 UWP 只是一個應用模型,本身是沒有什麼許可權管理的,許可權管理全靠 App Container 沙盒控制,如果我們脫離了這個沙盒,UWP 就會放飛自我了。那麼有沒... ...
  • 目錄條款17:讓介面容易被正確使用,不易被誤用(Make interfaces easy to use correctly and hard to use incorrectly)限制類型和值規定能做和不能做的事提供行為一致的介面條款19:設計class猶如設計type(Treat class de ...
  • title: 從零開始:Django項目的創建與配置指南 date: 2024/5/2 18:29:33 updated: 2024/5/2 18:29:33 categories: 後端開發 tags: Django WebDev Python ORM Security Deployment Op ...
  • 1、BOM對象 BOM:Broswer object model,即瀏覽器提供我們開發者在javascript用於操作瀏覽器的對象。 1.1、window對象 視窗方法 // BOM Browser object model 瀏覽器對象模型 // js中最大的一個對象.整個瀏覽器視窗出現的所有東西都 ...