博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
浅谈原型模式
阅读量:4937 次
发布时间:2019-06-11

本文共 3014 字,大约阅读时间需要 10 分钟。

      近期菜鸟D在看《大话设计模式》,看到原型模式,有一点自己的想法,所以就记下来。

      老规矩,解释定义:

      用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。

      Prototype原型模式是一种创建型设计模式,Prototype模式允许一个对象再创建另外一个可定制的对象,根本无需知道任何如何创建的细节,工作原理是:通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建。                          ——来自百度 

      由于感觉光看书不动手实践一下有点不靠谱,就按着自己的理解写了段代码,情景参照《大话设计模式》第9章,简历的复制。

代码如下:

//原型类实现接口ICloneablepublic class Prototype : ICloneable    {        public string ID { get; private set; }        public string Exp { get; set; }        private Company company = new Company();        public Prototype(string id)        {            ID = id;        }        public void SetExp(string exp)        {            Exp = exp;        }        #region 引用类型的拷贝        //SetCompany是浅拷贝        public void SetCompany(string address)        {            company.Address = address;        }        //SetCompany1是深拷贝        public void SetCompany1(string address)        {            Company c = (Company)company.Clone();            c.Address = address;            company = c;        }        #endregion        public void Show()        {            Console.WriteLine("id:{0} exp:{1} company:{2}", ID, Exp, company.Address);        }        public object Clone()        {            return MemberwiseClone();        }    }//公司类,也可以拷贝public class Company : ICloneable    {        private string address;        public string Address        {            get { return address; }            set { address = value; }        }        public object Clone()        {            return MemberwiseClone();        }    }//客户端            Prototype p1 = new Prototype("1");            p1.SetExp("半年经验");            p1.SetCompany("世纪财富中心");            //p1.Show();            Prototype p2 = (Prototype)p1.Clone();            p2.SetExp("1年经验");            p2.SetCompany("帝国大厦");            //p2.Show();            Prototype p3 = (Prototype)p1.Clone();            p3.SetCompany1("华尔街");            //p3.Show();            p1.Show();            p2.Show();            p3.Show();            //执行客户端的注释代码会发现有不同的结果,所以代码的执行逻辑也必须要理清

     运行结果:

结果1:

结果2:

结果3:

 

     代码的分析:

      在客户端代码中p2没有调用构造方法,而是调用p1的clone方法,所以内存中没有产生新的对象,p2引用p1的对象。当p2修改company时,p1的company也发生改变。p3同样调用p1的clone方法,但是p3使用SetCompany1来设置company,在SetCompany1的内部并不是直接给company赋值,而是重新克隆company,赋值给company字段,所以p3修改并没有将p1,p2的修改覆盖。出现运行结果1。

      在调整代码的执行顺序之后,可以看到p2的修改似乎没有影响p1(如结果2),但其实p2的修改已经影响了p1,只是p1的输出在前面执行。于是在结果3中可以看到修改。

      从以上代码及其结果,不难看出是深拷贝和浅拷贝以及代码的执行顺序的影响。

      原源码(大话设计模式P88)是通过在私有的构造函数中克隆WorkExperience,并将克隆体赋值给自己,从而产生区别。大话设计模式P87有深拷贝,浅拷贝的介绍。

      结合C++的知识,浅拷贝是将新对象的指针指向原有的对象,不再复制原有对象。深拷贝是将原有对象复制一个份,让新对象的指针指向复制后的地址。(如下附图)

      结合C#的知识,浅拷贝是新对象引用原有对象,内存中没有产生新的对象。深拷贝是在内存中开辟空间创建对象,并将原有对象的值复制,新对象引用内存中新产生的对象,只是与原有对象的值一致,从内存地址上看和原有对象没有关系。     

      应用的场景:

      从代码角度来说,避免创建对象时的初始化过程(如果这个过程占用的时间和资源都非常多),或者是希望避免使用工厂方法来实现多态的时候,可以考虑原型模式。

  从应用角度来说, 如果你创建的对象是多变化、多等级的产品,或者产品的创建过程非常耗时的时候(比如,有一定的计算量,或者对象创建时需要从网络或数据库中获取一定的数据),或者想把产品的创建独立出去,不想了解产品创建细节的时候可以考虑使用。

     注意事项:

     深拷贝和浅拷贝的使用场景,哪些是公有的允许修改的,哪些是不允许修改的。值类型和引用类型在深浅拷贝中的结果是不一样的。

     拷贝原型并进行修改意味着原型需要公开更多的数据,对已有系统实现原型模式可能修改的代价比较大。

附图:

部分内容摘自  

菜鸟D希望这篇文章对您有所帮助。

转载于:https://www.cnblogs.com/cnDqf/p/4108266.html

你可能感兴趣的文章
高级入门分界线
查看>>
S3C2440 中断相关寄存器小探
查看>>
Reverse Words in a String III
查看>>
高级软件工程课程的实践项目的自我目标
查看>>
Git常用操作命令
查看>>
智能合约从入门到精通:Lib工具库(一)
查看>>
面向对象与面向过程的简单比较
查看>>
手机端上下滑动选择项小组件
查看>>
Linux安装JDK
查看>>
[Asp.net 5] Configuration-新一代的配置文件(ConfigurationSource的多种实现)
查看>>
逗号运算符与逗号表达式
查看>>
vue学习demo
查看>>
display(block,inline,none),visibility(visible,hidden)之间的关系及区别
查看>>
hdu 1018:Big Number(水题)
查看>>
hrbustoj 1104:Leyni, LOLI and Line(解析几何,斜截式的应用)
查看>>
magento 搬家
查看>>
JS获取农历日期
查看>>
HDU 4267 A Simple Problem with Integers
查看>>
codeforces 154A 贪心
查看>>
循环神经网络(LSTM和GRU)(2)
查看>>