第 9 章 原型模式

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

原型模式其实就是从一个对象再创建另外一个可定制的对象,而且不需要知道任何创建的细节。

 

.Net在System命名空间中提供了ICloneable接口,其中就是唯一的一个方法Clone(),这样你就只需要实现这个接口就完成原型模式了。

public class Resume : ICloneable
    {
        private string sex;
        private string age;
        private string timeArea;
        private string company;

        //设置个人信息
        public void SetPersonalInfo(string sex, string age)
        {
            this.sex = sex;
            this.age = age;
        }

        //设置工作经历
        public void SetWorkExprience(string timeArea, string company)
        {
            this.timeArea = timeArea;
            this.company = company;
        }

        //显示
        public void Display()
        {
            Console.WriteLine("个人信息:{0} {1} {2}", name, sex, age);
            Console.WriteLine("工作经验:{0} {1}", timeArea, company);
        }

        //克隆
        public Object Clone()
        {
            return (Object)this.MemberwiseClone();
        }
    }

一般在初始化的信息不发生变化的情况下,克隆是最好的办法,这既隐瞒了对象创建的细节,又对性能是大大的提高。

不用重新初始化对象,而是动态的获得对象运行时的状态。

浅复制和深复制:

MemberwiseClone()方法是这样,如果字段是值类型的,则对该字段执行逐位复制。如果字段是引用类型,则复制引用但不复制引用的对象;因此,原始对象及其副本引用同一个对象。

代码结构变化:新建工作经验类

public class Resume : ICloneable
    {
        private string name;
        private string sex;
        private string age;
        private WorkExperience workExperience;
        public Resume(string name)
        {
            this.name = name;
            this.workExperience = new WorkExperience();
        }

        //设置个人信息
        public void SetPersonalInfo(string sex, string age)
        {
            this.sex = sex;
            this.age = age;
        }

        //设置工作经历
        public void SetWorkExprience(string timeArea, string company)
        {
            workExperience.WorkDate = timeArea;
            workExperience.Company = company;
        }

        //显示
        public void Display()
        {
            Console.WriteLine("个人信息:{0} {1} {2}", name, sex, age);
            Console.WriteLine("工作经验:{0} {1}", workExperience.WorkDate, workExperience.Company);
        }

        //克隆
        public Object Clone()
        {
            return (Object)this.MemberwiseClone();
        }
    }

    /// <summary>
    /// 工作经验类
    /// </summary>
    class WorkExperience
    {
        public string WorkDate { get; set; }
        public string Company { get; set; }
    }

客户端代码:

Resume r = new Resume("张三");
r.SetPersonalInfo("男", "25");
r.SetWorkExprience("2010", "嘻嘻");
var rr = (Resume)r.Clone();
rr.SetWorkExprience("2014", "xx嘻");

 r.Display();
 rr.Display();
 Console.ReadLine();

因为 MemberwiseClone只是浅表复制,所以结果为:

浅复制,被复制的对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用都仍然指向原来的对象。但是我们可能更需要这样的一个需求,把要复制的对象所引用的对象都复制一遍。

深复制:把引用对象的变量指向复制过来的新对象,而不是原来的被引用的对象。

改成深度复制:

 public class Resume : ICloneable
    {
        private string name;
        private string sex;
        private string age;
        private WorkExperience workExperience;
        public Resume(string name)
        {
            this.name = name;
            this.workExperience = new WorkExperience();
        }

        public Resume(WorkExperience workExperience)
        {
            this.workExperience = (WorkExperience)workExperience.Clone();
        }

        //设置个人信息
        public void SetPersonalInfo(string sex, string age)
        {
            this.sex = sex;
            this.age = age;
        }

        //设置工作经历
        public void SetWorkExprience(string timeArea, string company)
        {
            workExperience.WorkDate = timeArea;
            workExperience.Company = company;
        }

        //显示
        public void Display()
        {
            Console.WriteLine("个人信息:{0} {1} {2}", name, sex, age);
            Console.WriteLine("工作经验:{0} {1}", workExperience.WorkDate, workExperience.Company);
        }

        //克隆
        public Object Clone()
        {
            Resume r = new Resume(this.workExperience);
            r.name = this.name;
            r.sex = this.sex;
            r.age = this.age;
            return r;
        }
    }

    /// <summary>
    /// 工作经验类
    /// </summary>
    public class WorkExperience
    {
        public string WorkDate { get; set; }
        public string Company { get; set; }
        public Object Clone()
        {
            return (Object)this.MemberwiseClone();
        }
    }

结果:

时间: 2024-08-03 12:08:01

第 9 章 原型模式的相关文章

设计模式之第9章-原型模式(Java实现)

设计模式之第9章-原型模式(Java实现) “快到春节了,终于快放假了,天天上班好累的说.”“确实啊,最近加班比较严重,项目快到交付了啊.”“话说一到过节,就收到铺天盖地的短信轰炸,你说发短信就发吧,大多数还是一样的,群发.”“就是就是,上次我收到一个,竟然连名字都给弄错了,简直没法说啊,要不然就不发得了,干嘛弄得那么麻烦.”“所以说,才会有群发的短信我不回这个段子嘛.”“对了,今天你不是就要讲那个原型模式什么的,就是类似于群发的是吧.”“嘿嘿,天机不可泄露.”(PS:还天机不可泄露,学会吊起胃

大话设计模式C++实现-第9章-原型模式

一.UML图 二.概念 原型模式(Prototype):用原型示例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 三.说明 (1)原型模式实际上就是从一个对象再创建另外一个可定制的对象,而且不需要知道任何创建的细节. (2)一般在初始化的信息不发生变化的情况下,克隆是最好的办法.这既隐藏了对象创建的细节,有对性能是大大的提高.因为如果不用Clone,每次new,都需要执行一次构造函数,如果构造函数的执行时间很长,那么多次的执行这个初始化操作就实在是太低效了. (3)浅复制和深复制: 浅复

第九章 原型模式

大概是java版本升级了,clone方法不严格写好像也没什么问题. /** * Created by hero on 16-4-2. */ public class Company implements Cloneable { private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public Company(Stri

大话设计模式第九章---原型模式PHP实现

首先,PHP对象clone参考资料: http://php.net/manual/en/language.oop5.cloning.php#object.clone Object Cloning¶ Creating a copy of an object with fully replicated properties is not always the wanted behavior. A good example of the need for copy constructors, is i

「JavaScript里的面向对象」— 5.原型模式

本文原文来源:<Object-Oriented JavaScript>By Stoyan Stefanov 本文翻译来源:赤石俊哉 原创翻译 版权申明: 如果您是原文的原作者并且不希望此文被公开,可以联系作者删除.本文翻译由 赤石俊哉 翻译整理,您可以用于学习目的,但是禁止转载. 第五章 原型模式(Prototype) 在这一章节中你将会学习使用"函数(function)"对象中的prototype属性.在JavaScript的学习过程中,理解prototype的工作原理是

【设计模式+原型理解】第二章:基于构造函数扩展出来的原型模式

在第一章的时候,说过了单例模式.工厂模式.构造函数模式,你还记得构造模式是怎么样的吗? function CreateJsPerson(name, age) { this.name = name; this.age = age; this.writeJs = function() { console.log("my name is " + this.name + ", i can write js."); } } var p1 = new CreateJsPerso

第7章 创建型模式—原型模式

1. 原型模式(Prototype pattern)的定义 (1)用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象 ①通过克隆来创建新的对象实例 ②新的对象实例复制原型实例属性的值 (2)原型模式的结构和说明 ①Prototype:声明一个克隆自身的接口,用来约束想要克隆自己的类,要求他们都要实现这里定义的克隆方法. ②ConcretePrototype:实现Prototype接口的类,这些类真正实现了隆自身的功能. ③Client:使用原型的客户端,首先要获取到原型实例对象,然后

第6章 面向对象的程序设计 6.2 原型模式

我们创建的每个函数都有一个 prototype (原型)属性,这个属性是一个指针,指向一个对象,而这个对象的用途是包含特定类型的所有实例共享的属性和方法.按照字面意思来理解,那么 prototype 就是通过调用构造函数而创建的那个对象实例的原型对象.(有点类似于java中的类) 因此不必在构造函数中定义对象实例的信息,而是可以将这些信息直接添加到原型对象中,如下所示: function Person(){ } Person.prototype.name = "Nicholas"; P

javascript中的模式解析——原型模式

理解原型模式,首先要理解prototyoe(这个单词翻译 原型)属性,<javascript高级程序设计>书中描述到——我们创建的每个函数都有一个prototype属性,这个属性是一个指针,指向一个对象,而这个对象的用途是包含可以由特定类型的所有实例共享的属性和方法. 我在上一篇构造函数模式所说的,我们构建了一个一个Person函数,然后通过new一个person函数来创建了person1实例,person2实例,既然每个函数都有一个prototype属性,那么我有几个问题想要弄明白: per