简历复印--原型模式

1.1 夸张的简历

        简历的打印。"对编程来说,简单的复制粘贴极有可能造成重复代码的灾难。我所说的意思你根本还没听懂。那就以刚才的例子,我出个需求你写写看,要求有一个简历类,必须要有姓名,可以设置性别和年龄,可以设置工作经历。最终我需要写三份简历。"

1.2 简历代码初步实现

package code.chapter9.prototype1;public class Test {public static void main(String[] args){System.out.println("**********************************************");		System.out.println("《大话设计模式》代码样例");System.out.println();		Resume resume1 = new Resume("大鸟");resume1.setPersonalInfo("男","29");resume1.setWorkExperience("1998-2000","XX公司");Resume resume2 = new Resume("大鸟");resume2.setPersonalInfo("男","29");resume2.setWorkExperience("1998-2000","XX公司");Resume resume3 = new Resume("大鸟");resume3.setPersonalInfo("男","29");resume3.setWorkExperience("1998-2000","XX公司");resume1.display();resume2.display();resume3.display();System.out.println();System.out.println("**********************************************");}
}//简历类
class Resume  {private String name;private String sex;private String age;private String timeArea;private String company;public Resume(String name){this.name=name;}//设置个人信息public void setPersonalInfo(String sex,String age){this.sex=sex;this.age=age;}//设置工作经历public void setWorkExperience(String timeArea,String company){this.timeArea=timeArea;this.company=company;}//展示简历public void display(){System.out.println(this.name +" "+this.sex +" "+this.age);System.out.println("工作经历 "+this.timeArea +" "+this.company);}
}

        "很好,这其实就是当年我手写简历的时代的代码。三份简历需要三次实例化。你觉得这样的客户端代码是不是很麻烦?如果要二十份,你就需要二十次实例化。""是呀,而且如果我写错了一个字,比如1998年改成1999年,那就要改二十次。""你为什么不这样写呢?

package code.chapter9.prototype2;public class Test {public static void main(String[] args){System.out.println("**********************************************");		System.out.println("《大话设计模式》代码样例");System.out.println();		Resume resume1 = new Resume("大鸟");resume1.setPersonalInfo("男","29");resume1.setWorkExperience("1998-2000","XX公司");Resume resume2 = resume1;Resume resume3 = resume1;resume1.display();resume2.display();resume3.display();System.out.println();System.out.println("**********************************************");}
}//简历类
class Resume  {private String name;private String sex;private String age;private String timeArea;private String company;public Resume(String name){this.name=name;}//设置个人信息public void setPersonalInfo(String sex,String age){this.sex=sex;this.age=age;}//设置工作经历public void setWorkExperience(String timeArea,String company){this.timeArea=timeArea;this.company=company;}//展示简历public void display(){System.out.println(this.name +" "+this.sex +" "+this.age);System.out.println("工作经历 "+this.timeArea +" "+this.company);}
}

        "哈,这其实是传引用,而不是传值,这样做就如同是在resume2纸张和resume3纸张上写着简历在resume1处一样,没有实际的内容。"

1.3 原型模式

原型模式(Prototype),用原型实例指定创建对象的种类,并且通过复制这些原型创建新的对象。[DP]

原型模式(Prototype)结构图

"原型模式其实就是从一个对象再创建另外一个可定制的对象,而且不需要知道任何创建的细节。我们来看看基本的原型模式代码。"

package code.chapter9.prototype0;public class Test {public static void main(String[] args){System.out.println("**********************************************");		System.out.println("《大话设计模式》代码样例");System.out.println();		ConcretePrototype p1 = new ConcretePrototype("编号123456");System.out.println("原ID:"+ p1.getID());ConcretePrototype c1 = (ConcretePrototype)p1.clone();System.out.println("克隆ID:"+ c1.getID());System.out.println();System.out.println("**********************************************");}
}//原型类
abstract class Prototype implements Cloneable {private String id;public Prototype(String id){this.id=id;}public String getID(){return this.id;}//原型模式的关键就是有这样一个clone方法public Object clone(){Object object = null;try {object = super.clone();}catch(CloneNotSupportedException exception){System.err.println("Clone异常。");}return object;}
}//具体原型类
class ConcretePrototype extends Prototype{public ConcretePrototype(String id){super(id);}}

        "哦,这样就可以不用实例化ConcretePrototype了,直接克隆就行了?
        "说得没错,就是这样的。但对于Java而言,那个原型抽象类Prototype是用不着的,因为克隆实在是太常用了,所以Java提供了Cloneable接口,其中就是唯一的一个方法clone(),这样你就只需要实现这个接口就可以完成原型模式了。现在明白了?去改我们的'简历原型'代码吧。"

1.4 简历的原型实现

package code.chapter9.prototype3;public class Test {public static void main(String[] args){System.out.println("**********************************************");		System.out.println("《大话设计模式》代码样例");System.out.println();		Resume resume1 = new Resume("大鸟");resume1.setPersonalInfo("男","29");resume1.setWorkExperience("1998-2000","XX公司");Resume resume2 = resume1.clone();resume2.setWorkExperience("2000-2003","YY集团");Resume resume3 = resume1.clone();resume3.setPersonalInfo("男","24");resume1.display();resume2.display();resume3.display();System.out.println();System.out.println("**********************************************");}
}//简历类
class Resume implements Cloneable {private String name;private String sex;private String age;private String timeArea;private String company;public Resume(String name){this.name=name;}//设置个人信息public void setPersonalInfo(String sex,String age){this.sex=sex;this.age=age;}//设置工作经历public void setWorkExperience(String timeArea,String company){this.timeArea=timeArea;this.company=company;}//展示简历public void display(){System.out.println(this.name +" "+this.sex +" "+this.age);System.out.println("工作经历 "+this.timeArea +" "+this.company);}//实现了clone接口方法public Resume clone(){Resume object = null;try {object = (Resume)super.clone();}catch(CloneNotSupportedException exception){System.err.println("Clone异常。");}return object;}
}

        这样一来,客户端的代码就清爽很多了,而且你要是想改某份简历,只需要对这份简历做一定的修改就可以了,不会影响到其他简历,相同的部分就不用再重复了。不过不知道这样子对性能是不是有大的提高呢?"
        "当然是大大提高,你想呀,每new一次,都需要执行一次构造函数,如果构造函数的执行时间很长,那么多次执行这个初始化操作就实在太低效了。一般在初始化的信息不发生变化的情况下,克隆是最好的办法。这既隐藏了对象创建的细节,又对性能是大大的提高,何乐而不为呢?"
        "的确,我开始也没感觉到它的好,听你这么一说,感觉这样做的好处还真不少,它等于是不用重新初始化对象,而是动态地获得对象运行时的状态。这个模式真的很不错。

1.5 浅复制与深复制

        "别高兴得太早,如果我现在要改需求,你就又头疼了。你现在'简历'对象里的数据都是String型的,而String是一种拥有值类型特点的特殊引用类型,super.clone()方法是这样,如果字段是值类型的,则对该字段执行逐位复制,如果字段是引用类型,则复制引用但不复制引用的对象;因此,原始对象及其副本引用同一对象。什么意思呢?就是说如果你的'简历'类当中有对象引用,那么引用的对象数据是不会被克隆过来的。"
        "没太听懂,为什么不能一同复制过来呢?"

        "举个例子你就明白了,你现在的'简历'类当中有一个'设置工作经历'的方法,在现实设计当中,一般会再有一个'工作经历'类,当中有'时间区间'和'公司名称'等属性,'简历'类直接调用这个对象即可。

代码结构图

package code.chapter9.prototype4;public class Test {public static void main(String[] args){System.out.println("**********************************************");		System.out.println("《大话设计模式》代码样例");System.out.println();		Resume resume1 = new Resume("大鸟");resume1.setPersonalInfo("男","29");resume1.setWorkExperience("1998-2000","XX公司");Resume resume2 = resume1.clone();resume2.setWorkExperience("2000-2003","YY集团");Resume resume3 = resume1.clone();resume3.setPersonalInfo("男","24");resume3.setWorkExperience("2003-2006","ZZ公司");resume1.display();resume2.display();resume3.display();System.out.println();System.out.println("**********************************************");}
}//简历类
class Resume implements Cloneable {private String name;private String sex;private String age;private WorkExperience work;		//声明一个工作经历的对象public Resume(String name){this.name = name;this.work = new WorkExperience();//对这个工作经历对象实例化}//设置个人信息public void setPersonalInfo(String sex,String age){this.sex=sex;this.age=age;}//设置工作经历public void setWorkExperience(String timeArea,String company){this.work.setTimeArea(timeArea);//给工作经历实例的时间范围赋值this.work.setCompany(company);	//给工作经历实例的公司赋值}//展示简历public void display(){System.out.println(this.name +" "+this.sex +" "+this.age);System.out.println("工作经历 "+this.work.getTimeArea() +" "+this.work.getCompany());}public Resume clone(){Resume object = null;try {object = (Resume)super.clone();}catch(CloneNotSupportedException exception){System.err.println("Clone异常。");}return object;}
}//工作经历类
class WorkExperience {//工作时间范围private String timeArea;public String getTimeArea(){return this.timeArea;}public void setTimeArea(String value){this.timeArea=value;}//所在公司private String company;public String getCompany(){return this.company;}public void setCompany(String value){this.company=value;}
}

        结果显示,实际结果与期望结果并不符合,前两次的工作经历数据被最后一次数据给覆盖了。

        "通过写代码,并且去查了一下Java关于Cloneable的帮助,我大概知道你的意思了,由于它是浅表复制,所以对于值类型,没什么问题,对引用类型,就只是复制了引用,对引用的对象还是指向了原来的对象,所以就会出现我给resume1、resume2、resume3三个引用设置'工作经历',但却同时看到三个引用都是最后一次设置,因为三个引用都指向了同一个对象。"
        "你写的和说的都很好,就是这个原因,这叫作'浅复制',被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用都仍然指向原来的对象。但我们可能更需要这样的一种需求,把要复制的对象所引用的对象都复制一遍。比如刚才的例子,我们希望是resume1、resume2、resume3三个引用的对象是不同的,复制时就一变二,二变三,此时,我们就叫这种方式为'深复制',深复制把引用对象的变量指向复制过的新对象,而不是原有的被引用的对象。"
        "那如果'简历'对象引用了'工作经历','工作经历'再引用'公司','公司'再引用'职位'……这样一个引用一个,很多层,如何办?"
        "这的确是个很难回答的问题,深复制要深入到多少层,需要事先就考虑好,而且要当心出现循环引用的问题,需要小心处理,这里比较复杂,可以慢慢研究。就现在这个例子,问题应该不大,深入到第一层就可以了。"

1.6 简历的深复制实现

代码结构图

package code.chapter9.prototype5;public class Test {public static void main(String[] args){System.out.println("**********************************************");		System.out.println("《大话设计模式》代码样例");System.out.println();		Resume resume1 = new Resume("大鸟");resume1.setPersonalInfo("男","29");resume1.setWorkExperience("1998-2000","XX公司");Resume resume2 = resume1.clone();resume2.setWorkExperience("2000-2003","YY集团");Resume resume3 = resume1.clone();resume3.setPersonalInfo("男","24");resume3.setWorkExperience("2003-2006","ZZ公司");resume1.display();resume2.display();resume3.display();System.out.println();System.out.println("**********************************************");}
}//简历类
class Resume implements Cloneable {private String name;private String sex;private String age;private WorkExperience work;public Resume(String name){this.name = name;this.work = new WorkExperience();}//设置个人信息public void setPersonalInfo(String sex,String age){this.sex=sex;this.age=age;}//设置工作经历public void setWorkExperience(String timeArea,String company){this.work.setTimeArea(timeArea);//给工作经历实例的时间范围赋值this.work.setCompany(company);	//给工作经历实例的公司赋值}//展示简历public void display(){System.out.println(this.name +" "+this.sex +" "+this.age);System.out.println("工作经历 "+this.work.getTimeArea() +" "+this.work.getCompany());}public Resume clone(){Resume object = null;try {object = (Resume)super.clone();object.work = this.work.clone();}catch(CloneNotSupportedException exception){System.err.println("Clone异常。");}return object;}
}//工作经历类
class WorkExperience implements Cloneable {//工作时间范围private String timeArea;public String getTimeArea(){return this.timeArea;}public void setTimeArea(String value){this.timeArea=value;}//所在公司private String company;public String getCompany(){return this.company;}public void setCompany(String value){this.company=value;}public WorkExperience clone(){WorkExperience object = null;try {object = (WorkExperience)super.clone();}catch(CloneNotSupportedException exception){System.err.println("Clone异常。");}return object;}
}

1.7 复制简历vs.手写求职信

"如果是写代码,我当然会鼓励你去应用原型模式简化代码,优化设计。但对于求职,你是愿意你的简历和求职信倍受重视呢?还是愿意和所有的毕业生一样千篇一律毫无新意地碰运气?"

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/796808.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

web自动化测试系列-如何选择实战项目(四)

目录 1.项目安装 2.项目优点 3.项目介绍 3.1 项目介绍 3.3 商品展示 接上文 :web自动化测试系列-页面元素该如何定位?(三)-CSDN博客 在后续的技术操作中都将会使用到项目 ,如果你有自己的练手项目 ,优先拿自己的项目做练习&#xff0c…

【话题】如何看待那些速成并精通软件书籍的神器

大家好,我是全栈小5,欢迎阅读小5的系列文章,这是《话题》系列文章 目录 背景1. 神话与现实1.1 理论与实践之间的鸿沟1.2 一劳永逸的错觉 2. 速成书籍的优势与局限2.1 优势:2.2 局限: 3. 如何有效利用速成书籍3.1 量力而…

算法——链表(1)

T04BF 👋专栏: 算法|JAVA|MySQL|C语言 🫵 小比特 大梦想 此篇文章与大家分享链表专题的第一部分 如果有不足的或者错误的请您指出! 1.链表常用技巧总结 1.1引入虚拟头结点 在力扣上,基本提供的链表题目都是"无头的",但是针对无头链表,我们最…

【话题】程序员35岁会失业吗?

大家好,我是全栈小5,欢迎阅读小5的系列文章,这是《话题》系列文章 目录 背景招聘分析一、技术更新换代的挑战二、经验与技术的双重优势三、职业发展的多元化选择四、个人成长与职业规划的平衡五、结语文章推荐 背景 35岁被认为是程序员职业生…

数据治理与大数据平台解决方案(免费|可下载)

1、知识星球下载: 如需下载完整PPTX可编辑源文件,请前往星球获取:https://t.zsxq.com/191Y7dD3p 2、免费领取步骤: 【1】关注公众号 方案驿站 【2】私信发送 数据治理与大数据平台 【3】获取本方案PDF下载链接,直接…

007 CSS的继承和层叠 元素特性

文章目录 CSS属性的继承CSS属性的层叠选择器的权重 HTML元素的类型编写HTML注意事项元素隐藏方法CSS属性-overflowCSS样式不生效可能原因 CSS属性的继承 如果一个属性具备继承性,那么在该元素上设置后,它的后代元素都可以继承这个属性 如果后代元素自己…

2023年度总结:允许迷茫,破除迷茫;专注自身,把握当下

0、前言 📜为什么24年已经过了几个月,才提笔写这年度总结呢?毫不羞愧直问我的内心,其实就是懒罢了。直到前几天朋友看到了我去年写的总结,我自己点进那篇总结,完完整整的看了一遍,又翻看我23年…

C语言 | Leetcode C语言题解之第12题整数转罗马数字

题目: 题解: const char* thousands[] {"", "M", "MM", "MMM"}; const char* hundreds[] {"", "C", "CC", "CCC", "CD", "D", "DC"…

Vue2(完结):replace属性、编程式路由导航、缓存路由组件、两个新钩子、路由守卫、history与hash

一、router-link的replace属性 1、作用:控制路由跳转时操作浏览器历史记录的模式 2、浏览器的历史记录有两种写入方式:分别为push和replace,push是追加历史记录,replace是替换当前记录。路由跳转时候默认为push 3、如何开启repla…

敏感信息泄露漏洞

法律声明 参与培训需要遵守国家法律法规,相关知识只做技术研究,请勿用于违法用途,造成任何后果自负与本人无关。 中华人民共和国网络安全法(2017年6月1日起施行) 第二十二条 任何个人和组织不得从事入侵他人网络、干扰…

C——找单身狗2

题目内容: 在一个数组中,室友两个数字出现了一次,其他所有数字都出现了两次。找出只出现一次的数字。 如:1,2,3,4,5,1,2,3,4&#xff…

【云计算】云数据中心网络(一):VPC

云数据中心网络(一):VPC 1.什么是 VPC2.VPC 的组成2.1 虚拟交换机2.2 虚拟路由器 3.VPC 网络规划3.1 VPC 数量规划3.2 交换机数量规划3.3 地址空间规划3.4 不同规模企业地址空间规划实践 4.VPC 网络高可靠设计4.1 单地域单可用区部署4.2 单地…

【Vue3】自定义Modal的fixed无效

现有一个模态框&#xff0c;目前一切正常&#xff0c;Modal可以在视口正确位置展示 <template><div class"father"><h3>模态框的父组件</h3><button click"showModal !showModal">显示/隐藏modal</button><div …

NzN的C++之路--this指针

在已经入门了类和对象之后&#xff0c;今天我们来到了类和对象的part2部分--this指针。先三连后看是好习惯&#xff01;&#xff01;&#xff01; 目录 一、this指针的引入 二、this指针的特性 三、this指针必会 四、C语言和C实现Stack的对比 1. C语言实现 2. C实现 一、…

如何确认RID池是否耗尽,以及手动增加RID池大小

确认RID池是否耗尽&#xff1a; 事件查看器&#xff1a; 在RID主控域控制器上打开事件查看器&#xff0c;导航至“Windows日志 > 应用程序和服务日志 > Microsoft > Windows > Directory Service > Operations”。搜索事件ID 16656和16657。事件ID 16656表明RID…

Jupyterlab+内网云穿透傻瓜式教程

文章目录 Jupyterlab内网云穿透傻瓜式教程1、Miniforge安装2、Jupyter Lab安装3、Python语言服务器安装4、PowerShell 7安装5、更改jupyter lab配置6、内网穿透7、高级体验 Jupyterlab内网云穿透傻瓜式教程 1、Miniforge安装 如下图&#xff0c;以Windows安装为例&#xff0c…

DFS:floodfill算法解决矩阵联通块问题

floodfill&#xff0c;翻译为洪水灌溉&#xff0c;而floodfill算法本质上是为了解决在矩阵中性质相同的联通块问题。 一、图像渲染 . - 力扣&#xff08;LeetCode&#xff09; class Solution { public:int dx[4]{0,0,1,-1};int dy[4]{1,-1,0,0};int prev;//记住初始值int m,…

LangChain-10(2) 加餐 编写Agent获取本地Docker运行情况 无技术含量只是思路

可以先查看 上一节内容&#xff0c;会对本节有更好的理解。 安装依赖 pip install langchainhub编写代码 核心代码 tool def get_docker_info(docker_name: str) -> str:"""Get information about a docker pod container info."""result…

色彩在设计中的重要性

title: 色彩在设计中的重要性 date: 2024/4/6 19:08:21 updated: 2024/4/6 19:08:21 tags: 色彩心理品牌识别用户体验文化差异创意设计视觉传达易读性 色彩是设计中不可或缺的元素&#xff0c;它不仅可以影响人的情绪和心理状态&#xff0c;还可以在品牌识别、用户体验、文化差…

【C++练级之路】【Lv.19】【STL】unordered_set类和unordered_map类的模拟实现

快乐的流畅&#xff1a;个人主页 个人专栏&#xff1a;《算法神殿》《数据结构世界》《进击的C》 远方有一堆篝火&#xff0c;在为久候之人燃烧&#xff01; 文章目录 引言一、哈希表&#xff08;改造版&#xff09;1.1 结点1.2 迭代器1.2.1 operator 1.3 本体1.3.1 成员变量和…