深入理解Java設計模式之原型模式

一、前言

單例模式可以避免重復創建消耗資源的對象,但是卻不得不共用對象。若是對象本身也不讓隨意訪問修改時,怎麼辦?通常做法是備份到副本,其它對象操作副本,最後獲取權限合並,類似git上的PR操作。

二、什麼是原型模式

原型模式用原型實例指定創建對象的種類,並通過拷貝這些原型創建新的對象。需要註意的關鍵字是,新的對象,類沒變。.NET在System命名空間中提供瞭Cloneable接口,其中它提供唯一的方法Clone(),隻需要實現這個接口就可以完成原型模式瞭。由於它直接操作內存中的二進制流,當大量操作或操作復雜對象時,性能優勢將會很明顯。

三、原型模式的適用場景

多用於創建大對象,或初始化繁瑣的對象。如遊戲中的背景,地圖。web中的畫佈等等

以下場景適用:

一是類初始化需要消化非常多的資源,這個資源包括數據、硬件資源等;

二是通過 new 產生一個對象需要非常繁瑣的數據準備或訪問權限,則可以使用原型模式;

三是一個對象需要提供給其他對象訪問,而且各個調用者可能都需要修改其值時,可以考慮使用原型模式拷貝多個對象供調用者使用。

在實際項目中,原型模式很少單獨出現,一般是和工廠方法模式一起出現,通過 clone的方法創建一個對象,然後由工廠方法提供給調用者。

四、原型模式的實現

以簡歷的復印來舉例

1.淺拷貝實現

定義工作經歷類

/// <summary>
/// 工作經歷類
/// </summary>
public class WorkExperience
{
    private string _workDate;
    public string WorkDate
    {
        get { return _workDate; }
        set { _workDate = value; }
    }
     private string _company;
    public string Company
    {
        get { return _company; }
        set { _company = value; }
    }
}

定義簡歷類

/// <summary>
/// 簡歷類
/// </summary>
class Resume : ICloneable
{
    private string name;
    private string sex;
    private string age;
     private WorkExperience work;
     public Resume(string name)
    {
        this.name = name;
        work = new WorkExperience();
    }
     /// <summary>
    /// 設置個人信息
    /// </summary>
    /// <param name="sex"></param>
    /// <param name="age"></param>
    public void SetPersonalInfo(string sex, string age)
    {
        this.sex = sex;
        this.age = age;
    }
     /// <summary>
    /// 設置工作經歷
    /// </summary>
    /// <param name="workDate"></param>
    /// <param name="company"></param>
    public void SetWorkExperience(string workDate, string company)
    {
        work.WorkDate = workDate;
        work.Company = company;
    }
     /// <summary>
    /// 顯示
    /// </summary>
    public void Display()
    {
        Console.WriteLine("{0}{1}{2}", name, sex, age);
        Console.WriteLine("工作經歷:{0}{1}", work.WorkDate, work.Company);
    }
     public object Clone()
    {
        //創建當前object的淺表副本
        return (object)this.MemberwiseClone();
    }
}

客戶端調用

static void Main(string[] args)
{
    Resume a = new Resume("張三");
    a.SetPersonalInfo("男", "30");
    a.SetWorkExperience("2010-2018", "騰訊公司");
     Resume b = (Resume)a.Clone();
    b.SetWorkExperience("2010-2015", "阿裡公司");
     Resume c = (Resume)a.Clone();
    c.SetPersonalInfo("女", "18");  c.SetWorkExperience("2010-2015", "百度公司");
     a.Display();
    b.Display();
    c.Display();
     Console.Read();
}

結果

張三 男 30
工作經歷 2010-2018 騰訊公司
張三 男 30
工作經歷 2010-2018 騰訊公司
張三 女 18
工作經歷 2010-2018 騰訊公司

被復制對象的所有變量都含有與原來的對象相同的值,而所有的對其他對象的引用都仍然指向原來的對象,這就是淺復制。但是我們可能需要這樣一種需求,要把復制的對象所引用的對象都復制一遍。比如剛才的例子,我希望a、b、c三個引用的對象都是不同的。復制時就一變二,二變三。此時,我們就要用的方式叫“深復制”

2.深拷貝實現

深復制把引用對象的變量指向復制過的新對象,而不是原來被引用的對象

/// <summary>
/// 工作經歷類
/// </summary>
public class WorkExperience:ICloneable
{
    private string _workDate;
    public string WorkDate
    {
        get { return _workDate; }
        set { _workDate = value; }
    }
     private string _company;
    public string Company
    {
        get { return _company; }
        set { _company = value; }
    }
     public object Clone()
    {
        //創建當前object的淺表副本
        return (object)this.MemberwiseClone();
    }
}
/// <summary>
/// 簡歷類
/// </summary>
class Resume : ICloneable
{
    private string name;
    private string sex;
    private string age;
     private WorkExperience work;
     public Resume(string name)
    {
        this.name = name;
        work = new WorkExperience();
    }
     private Resume(WorkExperience work)
    {
        //提供Clone方法調用的私有構造函數,以便克隆“工作經歷”數據
        this.work = (WorkExperience)work.Clone();
    }
     /// <summary>
    /// 設置個人信息
    /// </summary>
    /// <param name="sex"></param>
    /// <param name="age"></param>
    public void SetPersonalInfo(string sex, string age)
    {
        this.sex = sex;
        this.age = age;
    }
     /// <summary>
    /// 設置工作經歷
    /// </summary>
    /// <param name="workDate"></param>
    /// <param name="company"></param>
    public void SetWorkExperience(string workDate, string company)
    {
        work.WorkDate = workDate;
        work.Company = company;
    }
     /// <summary>
    /// 顯示
    /// </summary>
    public void Display()
    {
        Console.WriteLine("{0}{1}{2}", name, sex, age);
        Console.WriteLine("工作經歷:{0}{1}", work.WorkDate, work.Company);
    }
     public object Clone()
    {
        //調用私有的構造方法,讓“工作經歷”克隆完成,然後再給這個簡歷對象的相關字段賦值,
        //最終返回一個深復制的簡歷對象
        Resume obj = new Resume(this.work);
        obj.name = this.name;
        obj.sex = this.sex;
        obj.age = this.age;
        return obj;
    }
}

客戶端調用代碼一樣

結果

張三 男 30
工作經歷 2010-2018 騰訊公司
張三 男 30
工作經歷 2010-2015 阿裡公司
張三 女 18
工作經歷 2010-2015 百度公司

由於在一些特定場合,會經常涉及深復制和淺復制,比如說,數據集對象DataSet,它就有Clone()方法和Copy()方法,Clone()方法用來復制DataSet的結構,但不復制DataSet的數據,實現瞭原型模式的淺復制,

Copy()方法不但復制結構,還復制數據,其實就是實現瞭原型模式的深復制。

五、總結

原型模式通過Object的clone()方法實現,由於是內存操作,無視構造方法和訪問權限,直接獲取新的對象。但對於引用類型,需使用深拷貝,其它淺拷貝即可。

推薦閱讀: