C# 設(shè)計(jì)模式系列教程-原型模式
1. 概述
通過復(fù)制一個(gè)已經(jīng)存在的實(shí)例來創(chuàng)建一個(gè)新的實(shí)例。被復(fù)制的實(shí)例被稱為原型,這個(gè)原型是可定制的。
2. 模式中的角色
2.1 抽象原型類(Abstract Prototype):提供一個(gè)克隆接口
2.2 具體原型類(Concrete Prototype): 及實(shí)現(xiàn)了克隆接口的具體原型類
3. 實(shí)例:求職網(wǎng)站上現(xiàn)在都支持多份簡(jiǎn)歷,如果每創(chuàng)建一份簡(jiǎn)歷都要從頭至尾地填寫一遍,那也是非常讓人沮喪的事。其實(shí)針對(duì)我們的求職崗位的不同,不同的簡(jiǎn)歷可能只要修改局部?jī)?nèi)容就可以了,而不用全部重新構(gòu)建一份新的簡(jiǎn)歷。復(fù)制一份簡(jiǎn)歷,然后做局部修改是最讓人省心的了!
3.1 實(shí)現(xiàn)類圖
類圖解讀
在.NET中,System命名空間已經(jīng)為我們提供了一個(gè)ICloneable接口,它包含了一個(gè)方法Clone(),實(shí)現(xiàn)這個(gè)接口就完成了原型模式。
3.2 在寫實(shí)現(xiàn)代碼之前,先要理解一下深復(fù)制與淺復(fù)制。
3.2.1 淺復(fù)制:將原來對(duì)象中的所有字段逐個(gè)復(fù)制到一個(gè)新對(duì)象,如果字段是值類型,則簡(jiǎn)單地復(fù)制一個(gè)副本到新對(duì)象,改變新對(duì)象的值類型字段不會(huì)影響原對(duì)象;如果字段是引用類型,則復(fù)制的是引用,改變目標(biāo)對(duì)象中引用類型字段的值將會(huì)影響原對(duì)象。例如, 如果一個(gè)對(duì)象有一個(gè)指向引用類型(如例子中的工作經(jīng)歷)的字段, 并且我們對(duì)該對(duì)象做了一個(gè)淺復(fù)制, 那麼兩個(gè)對(duì)象將引用同一個(gè)引用(即同一段工作經(jīng)歷)。
3.2.2 深復(fù)制:與淺復(fù)制不同之處在于對(duì)引用類型的處理,深復(fù)制將新對(duì)象中引用類型字段指向復(fù)制過的新對(duì)象,改變新對(duì)象中引用的任何對(duì)象,不會(huì)影響到原來的對(duì)象中對(duì)應(yīng)字段的內(nèi)容。例如,如果一個(gè)對(duì)象有一個(gè)指向引用類型(如例子中的工作經(jīng)歷)的字段,并且對(duì)該對(duì)象做了一個(gè)深復(fù)制的話.我門將創(chuàng)建一個(gè)新的對(duì)象(即新的工作經(jīng)歷)。
3.3 簡(jiǎn)歷的淺復(fù)制實(shí)現(xiàn)
/// <summary>
/// 實(shí)現(xiàn)了ICloneable接口的簡(jiǎn)歷類
/// </summary>
public class Resume:ICloneable
{
public Resume()
{
mWorkExperience = new WorkExperience();
}
private string mName;
private string mSex;
private int mAge;
private WorkExperience mWorkExperience;
public string Name
{
get { return mName; }
set { mName = value; }
}
public string Sex
{
get { return mSex; }
set { mSex = value; }
}
public int Age
{
get { return mAge; }
set { mAge = value; }
}
/// <summary>
/// 關(guān)聯(lián)了一個(gè)引用類型
/// </summary>
public WorkExperience WorkExperience
{
get { return mWorkExperience; }
}
public void SetWorkExperience(DateTime startDate, DateTime endDate, string company, string position)
{
this.mWorkExperience.Company = company;
this.mWorkExperience.EndDate = endDate;
this.mWorkExperience.StartDate = startDate;
this.mWorkExperience.Position = position;
}
/// <summary>
/// 實(shí)現(xiàn)ICloneable接口的Clone方法
/// </summary>
/// <returns></returns>
public object Clone()
{
// .Net 為我們提供的淺復(fù)制對(duì)象的方法
return this.MemberwiseClone();
}
}
/// <summary>
/// 工作經(jīng)歷類
/// </summary>
public class WorkExperience
{
public DateTime StartDate { get; set; }
public DateTime EndDate { get; set; }
public string Company { get; set; }
public string Position { get; set; }
}
下面是測(cè)試代碼
[TestMethod]
public void TestShallowCopy()
{
Resume myFirstResume = new Resume
{
Age = 29,
Name = "Kevin Wang",
Sex = "男",
};
myFirstResume.SetWorkExperience(new DateTime(2006, 7, 1), new DateTime(2007, 7, 1), "My First Company", "Software Engineer");
Resume mySecondResume = (Resume)myFirstResume.Clone();
mySecondResume.SetWorkExperience(new DateTime(2007, 8, 1), new DateTime(2008, 8, 1), "My Second Company", "Software Engineer");
Resume myThirdResume = (Resume)myFirstResume.Clone();
myThirdResume.SetWorkExperience(new DateTime(2008, 8, 1), new DateTime(2009, 8, 1), "My Third Company", "Senior Software Engineer");
Assert.AreEqual("My First Company", myFirstResume.WorkExperience.Company);
Assert.AreEqual("My Second Company", mySecondResume.WorkExperience.Company);
Assert.AreEqual("My Third Company", myThirdResume.WorkExperience.Company);
}
這里期望的是三個(gè)斷言都能運(yùn)行成功,但是卻是失敗的,原因是:由于我們使用的是淺復(fù)制,所以myFirstResume, mySecondResume 和 myThirdResume引用的是同一個(gè)對(duì)象,因此最終的結(jié)果是 三個(gè)簡(jiǎn)歷的WorkExperience.Company都是“My Third Company".
3.4 簡(jiǎn)歷的深復(fù)制實(shí)現(xiàn)
/// <summary>
/// 實(shí)現(xiàn)了ICloneable接口的簡(jiǎn)歷類
/// </summary>
public class Resume : ICloneable
{
public Resume()
{
mWorkExperience = new WorkExperience();
}
/// <summary>
/// 這里使用一個(gè)私有的構(gòu)造函數(shù)來對(duì)其連接到的引用類型進(jìn)行復(fù)制
/// </summary>
/// <param name="workExperience"></param>
private Resume(WorkExperience workExperience)
{
this.mWorkExperience = (WorkExperience)workExperience.Clone();
}
private string mName;
private string mSex;
private int mAge;
private WorkExperience mWorkExperience;
public string Name
{
get { return mName; }
set { mName = value; }
}
public string Sex
{
get { return mSex; }
set { mSex = value; }
}
public int Age
{
get { return mAge; }
set { mAge = value; }
}
public WorkExperience WorkExperience
{
get { return mWorkExperience; }
}
/// <summary>
/// 設(shè)置功過經(jīng)歷
/// </summary>
/// <param name="startDate"></param>
/// <param name="endDate"></param>
/// <param name="company"></param>
/// <param name="position"></param>
public void SetWorkExperience(DateTime startDate, DateTime endDate, string company, string position)
{
this.mWorkExperience.Company = company;
this.mWorkExperience.EndDate = endDate;
this.mWorkExperience.StartDate = startDate;
this.mWorkExperience.Position = position;
}
/// <summary>
/// 實(shí)現(xiàn)ICloneable接口的Clone方法
/// </summary>
/// <returns></returns>
public object Clone()
{
// 這里不再使用MemberwiseClone方法進(jìn)行復(fù)制了,而是新創(chuàng)建了一個(gè)全新的簡(jiǎn)歷。它完全是在內(nèi)部實(shí)現(xiàn)的,外部不用關(guān)心它的實(shí)現(xiàn)
Resume newResume = new Resume(this.mWorkExperience);
newResume.mSex = this.mSex;
newResume.mName = this.mName;
newResume.mAge = this.mAge;
return newResume;
}
}
public class WorkExperience :ICloneable
{
public DateTime StartDate { get; set; }
public DateTime EndDate { get; set; }
public string Company { get; set; }
public string Position { get; set; }
public object Clone()
{
// 使用.Net 為我們提供的淺復(fù)制對(duì)象的方法,因?yàn)檫@里已經(jīng)沒有引用對(duì)象了(string雖然是引用類型,但.NET為我們做了特別處理,可以像值類型一樣使用它)。
return this.MemberwiseClone();
}
}
測(cè)試代碼如下
[TestMethod]
public void TestDeepCopy()
{
Resume myFirstResume = new Resume
{
Age = 29,
Name = "Kevin Wang",
Sex = "男",
};
myFirstResume.SetWorkExperience(new DateTime(2006, 7, 1), new DateTime(2007, 7, 1), "My First Company", "Software Engineer");
Resume mySecondResume = (Resume)myFirstResume.Clone();
mySecondResume.SetWorkExperience(new DateTime(2007, 8, 1), new DateTime(2008, 8, 1), "My Second Company", "Software Engineer");
Resume myThirdResume = (Resume)myFirstResume.Clone();
myThirdResume.SetWorkExperience(new DateTime(2008, 8, 1), new DateTime(2009, 8, 1), "My Third Company", "Senior Software Engineer");
Assert.AreEqual("My First Company", myFirstResume.WorkExperience.Company);
Assert.AreEqual("My Second Company", mySecondResume.WorkExperience.Company);
Assert.AreEqual("My Third Company", myThirdResume.WorkExperience.Company);
}
運(yùn)行測(cè)試,測(cè)試通過,這正是我們期望的結(jié)果。
4. 模式總結(jié)
4.1 優(yōu)點(diǎn)
4.1.1 隱藏了對(duì)象的創(chuàng)建細(xì)節(jié),對(duì)有些初始化需要占用很多資源的類來說,對(duì)性能也有很大提高。
4.1.2 在需要新對(duì)象時(shí),可以使用Clone來快速創(chuàng)建創(chuàng)建一個(gè),而不用使用new來構(gòu)建。
4.2 缺點(diǎn)
4.2.1 每一個(gè)類都需要一個(gè)Clone方法,而且必須通盤考慮。對(duì)于深拷貝來說,每個(gè)關(guān)聯(lián)到的類型都不許實(shí)現(xiàn)IClonable接口,并且每增加或修改一個(gè)字段是都需要更新Clone方法。
4.3 適用場(chǎng)景
4.3.1 類初始化需要消化非常多的資源,這個(gè)資源包括數(shù)據(jù)、硬件資源等
4.3.2 通過new產(chǎn)生一個(gè)對(duì)象需要非常繁瑣的數(shù)據(jù)準(zhǔn)備或訪問權(quán)限,則可以使用原型模式
4.3.3 一個(gè)對(duì)象需要提供給其他對(duì)象訪問,而且各個(gè)調(diào)用者可能都需要修改其值時(shí),可以考慮使用原型模式拷貝多個(gè)對(duì)象供調(diào)用者使用。
以上就是本文的全部?jī)?nèi)容,希望能給大家一個(gè)參考,也希望大家多多支持我們。
上一篇:vs2005中總是保留最近打開的項(xiàng)目和文件的記錄
欄 目:C#教程
本文標(biāo)題:C# 設(shè)計(jì)模式系列教程-原型模式
本文地址:http://www.jygsgssxh.com/a1/C_jiaocheng/6468.html
您可能感興趣的文章
- 01-10深入淺出23種設(shè)計(jì)模式
- 01-10TortoiseSVN使用教程
- 01-10C#編程中枚舉類型的使用教程
- 01-10Python設(shè)計(jì)模式編程中的備忘錄模式與對(duì)象池模式示例
- 01-10C#中的delegate委托類型基本學(xué)習(xí)教程
- 01-10dotNet中的反射用法入門教程
- 01-10詳解C#的設(shè)計(jì)模式編程之抽象工廠模式的應(yīng)用
- 01-10解析C#設(shè)計(jì)模式編程中的裝飾者模式
- 01-10簡(jiǎn)單了解C#設(shè)計(jì)模式編程中的橋接模式
- 01-10C#編程中使用設(shè)計(jì)模式中的原型模式的實(shí)例講解


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-10C#通過反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁無法打開的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
- 01-10WinForm實(shí)現(xiàn)仿視頻播放器左下角滾動(dòng)新
- 01-10C#停止線程的方法
- 01-10C#實(shí)現(xiàn)清空回收站的方法
- 01-10C#通過重寫Panel改變邊框顏色與寬度的
- 01-10C#實(shí)現(xiàn)讀取注冊(cè)表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改


