服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C# 设计模式系列教程-原型模式

C# 设计模式系列教程-原型模式

2021-11-23 15:13Wang Juqiang C#

原型模式隐藏了对象的创建细节,对有些初始化需要占用很多资源的类来说,对性能也有很大提高。

1. 概述

  通过复制一个已经存在的实例来创建一个新的实例。被复制的实例被称为原型,这个原型是可定制的。

2. 模式中的角色

  2.1 抽象原型类(abstract prototype):提供一个克隆接口

  2.2 具体原型类(concrete prototype): 及实现了克隆接口的具体原型类

3. 实例:求职网站上现在都支持多份简历,如果每创建一份简历都要从头至尾地填写一遍,那也是非常让人沮丧的事。其实针对我们的求职岗位的不同,不同的简历可能只要修改局部内容就可以了,而不用全部重新构建一份新的简历。复制一份简历,然后做局部修改是最让人省心的了!

  3.1 实现类图

C# 设计模式系列教程-原型模式

  类图解读

  在.net中,system命名空间已经为我们提供了一个icloneable接口,它包含了一个方法clone(),实现这个接口就完成了原型模式。

  3.2 在写实现代码之前,先要理解一下深复制与浅复制。

    3.2.1 浅复制:将原来对象中的所有字段逐个复制到一个新对象,如果字段是值类型,则简单地复制一个副本到新对象,改变新对象的值类型字段不会影响原对象;如果字段是引用类型,则复制的是引用,改变目标对象中引用类型字段的值将会影响原对象。例如, 如果一个对象有一个指向引用类型(如例子中的工作经历)的字段, 并且我们对该对象做了一个浅复制, 那麽两个对象将引用同一个引用(即同一段工作经历)。

    3.2.2 深复制:与浅复制不同之处在于对引用类型的处理,深复制将新对象中引用类型字段指向复制过的新对象,改变新对象中引用的任何对象,不会影响到原来的对象中对应字段的内容。例如,如果一个对象有一个指向引用类型(如例子中的工作经历)的字段,并且对该对象做了一个深复制的话.我门将创建一个新的对象(即新的工作经历)。

  3.3 简历的浅复制实现

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/// <summary>
/// 实现了icloneable接口的简历类
/// </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>
/// 关联了一个引用类型
/// </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>
/// 实现icloneable接口的clone方法
/// </summary>
/// <returns></returns>
public object clone()
{
 // .net 为我们提供的浅复制对象的方法
 return this.memberwiseclone();
}
}
 
/// <summary>
/// 工作经历类
/// </summary>
public class workexperience
{
public datetime startdate { get; set; }
public datetime enddate { get; set; }
public string company { get; set; }
public string position { get; set; }
}

 

  下面是测试代码

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
[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);
}

 

    这里期望的是三个断言都能运行成功,但是却是失败的,原因是:由于我们使用的是浅复制,所以myfirstresume, mysecondresume 和 mythirdresume引用的是同一个对象,因此最终的结果是 三个简历的workexperience.company都是“my third company".

  3.4 简历的深复制实现

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
/// <summary>
/// 实现了icloneable接口的简历类
/// </summary>
public class resume : icloneable
{
public resume()
{
 mworkexperience = new workexperience();
}
 
/// <summary>
/// 这里使用一个私有的构造函数来对其连接到的引用类型进行复制
/// </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>
/// 设置功过经历
/// </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>
/// 实现icloneable接口的clone方法
/// </summary>
/// <returns></returns>
public object clone()
{
 // 这里不再使用memberwiseclone方法进行复制了,而是新创建了一个全新的简历。它完全是在内部实现的,外部不用关心它的实现
 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 为我们提供的浅复制对象的方法,因为这里已经没有引用对象了(string虽然是引用类型,但.net为我们做了特别处理,可以像值类型一样使用它)。
 return this.memberwiseclone();
}
}

 

  测试代码如下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
[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);
}

 

  运行测试,测试通过,这正是我们期望的结果。

4. 模式总结

  4.1 优点

    4.1.1 隐藏了对象的创建细节,对有些初始化需要占用很多资源的类来说,对性能也有很大提高。

    4.1.2 在需要新对象时,可以使用clone来快速创建创建一个,而不用使用new来构建。

  4.2 缺点

    4.2.1 每一个类都需要一个clone方法,而且必须通盘考虑。对于深拷贝来说,每个关联到的类型都不许实现iclonable接口,并且每增加或修改一个字段是都需要更新clone方法。

  4.3 适用场景

    4.3.1 类初始化需要消化非常多的资源,这个资源包括数据、硬件资源等

    4.3.2 通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式

    4.3.3 一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用。

以上就是本文的全部内容,希望能给大家一个参考,也希望大家多多支持服务器之家。

延伸 · 阅读

精彩推荐
  • C#C#微信公众号与订阅号接口开发示例代码

    C#微信公众号与订阅号接口开发示例代码

    这篇文章主要介绍了C#微信公众号与订阅号接口开发示例代码,结合实例形式简单分析了C#针对微信接口的调用与处理技巧,需要的朋友可以参考下...

    smartsmile20127762021-11-25
  • C#深入理解C#的数组

    深入理解C#的数组

    本篇文章主要介绍了C#的数组,数组是一种数据结构,详细的介绍了数组的声明和访问等,有兴趣的可以了解一下。...

    佳园9492021-12-10
  • C#C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    这篇文章主要介绍了C#设计模式之Strategy策略模式解决007大破密码危机问题,简单描述了策略模式的定义并结合加密解密算法实例分析了C#策略模式的具体使用...

    GhostRider10972022-01-21
  • C#VS2012 程序打包部署图文详解

    VS2012 程序打包部署图文详解

    VS2012虽然没有集成打包工具,但它为我们提供了下载的端口,需要我们手动安装一个插件InstallShield。网上有很多第三方的打包工具,但为什么偏要使用微软...

    张信秀7712021-12-15
  • C#SQLite在C#中的安装与操作技巧

    SQLite在C#中的安装与操作技巧

    SQLite,是一款轻型的数据库,用于本地的数据储存。其优点有很多,下面通过本文给大家介绍SQLite在C#中的安装与操作技巧,感兴趣的的朋友参考下吧...

    蓝曈魅11162022-01-20
  • C#如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    这篇文章主要给大家介绍了关于如何使用C#将Tensorflow训练的.pb文件用在生产环境的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴...

    bbird201811792022-03-05
  • C#利用C#实现网络爬虫

    利用C#实现网络爬虫

    这篇文章主要介绍了利用C#实现网络爬虫,完整的介绍了C#实现网络爬虫详细过程,感兴趣的小伙伴们可以参考一下...

    C#教程网11852021-11-16
  • C#三十分钟快速掌握C# 6.0知识点

    三十分钟快速掌握C# 6.0知识点

    这篇文章主要介绍了C# 6.0的相关知识点,文中介绍的非常详细,通过这篇文字可以让大家在三十分钟内快速的掌握C# 6.0,需要的朋友可以参考借鉴,下面来...

    雨夜潇湘8272021-12-28