第 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(); } }
结果:
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 一个奇形怪状的面试题:Bean中的CHM要不要加volatile?
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· Obsidian + DeepSeek:免费 AI 助力你的知识管理,让你的笔记飞起来!
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了