設計模式學習心得<原型模式 Prototype >

原型模式(Prototype Pattern)是用於建立重複的對象,同時又能保證性能。這種類型的設計模式屬於建立型模式,它提供了一種建立對象的最佳方式。數據庫

這種模式是實現了一個原型接口,該接口用於建立當前對象的克隆。當直接建立對象的代價比較大時,則採用這種模式。例如,一個對象須要在一個高代價的數據庫操做以後被建立。咱們能夠緩存該對象,在下一個請求時返回它的克隆,在須要的時候更新數據庫,以此來減小數據庫調用。設計模式

如今想來,實際開發過程當中極少應用這一模式,在於Spring已經幫咱們實現了。
當咱們須要調用對象時,經常只需@Autowired緩存

概述


  • 意圖
    用原型實例指定建立對象的種類,而且經過拷貝這些原型建立新的對象。安全

  • 主要解決
    在運行期創建和刪除原型。ide

  • 什麼時候使用
  1. 當一個系統應該獨立於它的產品建立,構成和表示時。
  2. 當要實例化的類是在運行時刻指定時,例如,經過動態裝載。
  3. 爲了不建立一個與產品類層次平行的工廠類層次時。
  4. 當一個類的實例只能有幾個不一樣狀態組合中的一種時。創建相應數目的原型並克隆它們可能比每次用合適的狀態手工實例化該類更方便一些。
  • 如何解決
    利用已有的一個原型對象,快速地生成和原型對象同樣的實例。函數

  • 關鍵代碼
  1. 實現克隆操做,在 JAVA 繼承 Cloneable,重寫 clone(),在 .NET 中可使用 Object 類的 MemberwiseClone() 方法來實現對象的淺拷貝或經過序列化的方式來實現深拷貝。
  2. 原型模式一樣用於隔離類對象的使用者和具體類型(易變類)之間的耦合關係,它一樣要求這些"易變類"擁有穩定的接口。
  • 應用實例
  1. 細胞分裂。
  2. JAVA 中的 Object clone() 方法。
  • 優勢
  1. 性能提升。
  2. 逃避構造函數的約束。
  • 缺點
  1. 配備克隆方法須要對類的功能進行通盤考慮,這對於全新的類不是很難,但對於已有的類不必定很容易,特別當一個類引用不支持串行化的間接對象,或者引用含有循環結構的時候。
  2. 必須實現 Cloneable接口。
  • 使用場景
  1. 資源優化場景。
  2. 類初始化須要消化很是多的資源,這個資源包括數據、硬件資源等。
  3. 性能和安全要求的場景。例:拷貝文件,對鏡像進行修改。當修改完成替換原有文件。
  4. 經過 new 產生一個對象須要很是繁瑣的數據準備或訪問權限,則可使用原型模式。
  5. 一個對象多個修改者的場景。
  6. 一個對象須要提供給其餘對象訪問,並且各個調用者可能都須要修改其值時,能夠考慮使用原型模式拷貝多個對象供調用者使用。
  7. 在實際項目中,原型模式不多單獨出現,通常是和工廠方法模式一塊兒出現,經過 clone 的方法建立一個對象,而後由工廠方法提供給調用者。原型模式已經與 Java 融爲渾然一體,你們能夠隨手拿來使用。
  8. 在事務處理場景中,@transactional方法還沒有結束,又想對因執行save、update等方法的對象,進行修改。

注意事項:與經過對一個類進行實例化來構造新對象不一樣的是,原型模式是經過拷貝一個現有對象生成新對象的。淺拷貝實現 Cloneable,重寫,深拷貝是經過實現 Serializable讀取二進制流。性能

實現


public class Message implements Cloneable{
    private Integer id;
    private String phone;
    private String token;
    private String verificationCode;
    private Date verifiedAt;
    private String ipAddress;
    private Date createdAt;
    private Date updatedAt;

    public Message() {
    }

    public Message(String phone, String ipAddress, String token) {
        this.phone = phone;
        this.ipAddress = ipAddress;
        this.token = token;
    }

    /**
    * 利用GenerateCopyConstructor插件生成
    */
    public Message(Message other) {
        this.id = other.id;
        this.phone = other.phone;
        this.token = other.token;
        this.verificationCode = other.verificationCode;
        this.verifiedAt = other.verifiedAt;
        this.ipAddress = other.ipAddress;
        this.createdAt = other.createdAt;
        this.updatedAt = other.updatedAt;
    }

    @Override
    public Message clone() {
        return new Message(this);
    }
    
    public Message(String phone, String ipAddress, String token, String verificationCode) {
        this.phone = phone;
        this.ipAddress = ipAddress;
        this.token = token;
        this.verificationCode = verificationCode;
    }

    public Integer getId() {
        return id;
    }

    public void setId(Integer id) {
        this.id = id;
    }

    .......

}

提示:若是是使用idea,能夠利用GenerateCopyConstructor插件,協助生成
(在此略去插件安裝過程)優化

  1. 選中代碼中任意位置(方法定義以外),點擊右鍵
  2. 點擊Generate...,Mac 快捷鍵Commond + n,Windows快捷鍵Alt + Insert
  3. 點擊Copy Constructor

利用BeanUtils實現this

相關文章
相關標籤/搜索