给女友讲讲设计模式——原型模式(JAVA实例)4
前言
上高中那阵子,特别迷恋火影,所以每周都会等着更新。最让我难忘的是里面的主人公鸣人,和他的各种各样炫酷的技能。他的影分身,一下子可以分离出来多个自己,并且这些自己的副本使用的技能都是相同的。如果他的副本想要使用其他的技能,除非分身再克隆出来一个技能的副本,然后去进行学习。
正文
刚刚说的那个例子,大家先记着,接下来我慢慢揭开圆形模式的面纱。
原型模式,简单来说就像是我们操纵电脑使的复制粘贴,只不过我们是要去操纵对象。在java中使用原型模式特别的简单,只需要重写clone()方法,然后实现Cloneable接口(其实这个接口是一个空接口,只是为了告诉jvm,这个类是可以克隆的);
原型模式一共分为两种:
1.浅克隆
2.深克隆
下面看看浅克隆的代码
这是一个忍者的实体类,代表鸣人的。
package prototype; public class Nanjia implements Cloneable{ private String name; private Integer age; private Skill skill; public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public Skill getSkill() { return skill; } public void setSkill(Skill skill) { this.skill = skill; } @Override public Nanjia clone() { try { Nanjia nanjia=(Nanjia)super.clone(); return nanjia; } catch (CloneNotSupportedException e) { // TODO Auto-generated catch block e.printStackTrace(); } return null; } @Override public String toString() { return "Nanjia [name=" + name + ", age=" + age + ", skill=" + skill + "]"; } }
然后是一个skill的实体类,可以装者技能的种类和名称
package prototype; public class Skill{ private String name; private String type; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getType() { return type; } public void setType(String type) { this.type = type; } @Override public String toString() { return "Skill [name=" + name + ", type=" + type + "]"; } }
package prototype; public class MainTest { public static void main(String[] args) { Nanjia nanjia=new Nanjia(); nanjia.setAge(16); nanjia.setName("漩涡鸣人"); Skill skill=new Skill(); skill.setName("仙人模式"); nanjia.setSkill(skill); System.out.println("这是需要被克隆的类"); System.out.println(nanjia); Nanjia nanjia2=nanjia.clone(); Skill skill2=nanjia2.getSkill(); skill2.setName("螺旋丸"); nanjia2.setSkill(skill2); System.out.println("这是克隆出来的类"); System.out.println(nanjia2); System.out.println("这是原来的类"); System.out.println(nanjia); } }
在测试类中我们做了这样的操作,将一些属性赋予nanji的实体类中,然后我们通过克隆复制出了一个副本,可以看作类似于影分身一样,然后拿出副本中的技能将它转换为别的技能,最后我们会发现原来本身的技能也发生了变化,这是怎么一回事呢?
这是因为浅克隆,在克隆的过程中,对象中的实体引用,只是复制了这个引用的地址,所以在修改的时候就相当于把原来本体的技能同样进行修改。
那我们要是想不影响之前的本体该怎么办呢,接下来隆重介绍深克隆。
深克隆是把忍者类中的技能类也重写了克隆方法,这样在影分身的副本中我们就可以自由转换,而不影响忍者这个本体了。
然后是深克隆的代码:
package prototype; public class Nanjia implements Cloneable{ private String name; private Integer age; private Skill skill; public String getName() { return name; } public void setName(String name) { this.name = name; } public int getAge() { return age; } public void setAge(int age) { this.age = age; } public Skill getSkill() { return skill; } public void setSkill(Skill skill) { this.skill = skill; } @Override public Nanjia clone() { try { Nanjia nanjia=(Nanjia)super.clone(); //关键的地方在这里,我们像克隆忍者一样,把技能这个类也进行了克隆 nanjia.setSkill((Skill)nanjia.getSkill().clone()); return nanjia; } catch (CloneNotSupportedException e) { // TODO Auto-generated catch block e.printStackTrace(); } return null; } @Override public String toString() { return "Nanjia [name=" + name + ", age=" + age + ", skill=" + skill + "]"; } }
我们继承了Cloneable接口,重写了Clone这个方法。
package prototype; public class Skill implements Cloneable{ private String name; private String type; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getType() { return type; } public void setType(String type) { this.type = type; } @Override public String toString() { return "Skill [name=" + name + ", type=" + type + "]"; } @Override public Object clone() { try { return super.clone(); } catch (CloneNotSupportedException e) { e.printStackTrace(); return null; } } }
package prototype; public class MainTest { public static void main(String[] args) { Nanjia nanjia=new Nanjia(); nanjia.setAge(16); nanjia.setName("漩涡鸣人"); Skill skill=new Skill(); skill.setName("仙人模式"); nanjia.setSkill(skill); System.out.println("这是需要被克隆的类"); System.out.println(nanjia); Nanjia nanjia2=nanjia.clone(); Skill skill2=nanjia2.getSkill(); skill2.setName("螺旋丸"); nanjia2.setSkill(skill2); System.out.println("这是克隆出来的类"); System.out.println(nanjia2); System.out.println("这是原来的类"); System.out.println(nanjia); } }
后记
克隆模式却是在制作副本,并且副本极其相似的情况下,给我们带来了很多便利性,但是如果类结构相对复杂,我们不但得在每一个实体中加入克隆方法,并且这样做还不符合开闭原则。所以在使用之前,应该根据需求,看看需不需要用到原型模式
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
程序员面试系列之Java单例模式的攻击与防御
我写的程序员面试系列 Java面试系列-webapp文件夹和WebContent文件夹的区别? 程序员面试系列:Spring MVC能响应HTTP请求的原因? Java程序员面试系列-什么是Java Marker Interface(标记接口) 使用JDK自带的工具jstack找出造成运行程序死锁的原因 编程面试题:编写一个会造成数据库死锁的应用 JavaScript面试系列:JavaScript设计模式之桥接模式和懒加载 使用JavaScript ES6的新特性计算Fibonacci(非波拉契数列) 单例模式在很多Java程序员的眼中,应该是设计模式里最简单的一种了。那么单例模式可能会被攻击,您听说过么? 说到“单例模式被攻击”这个话题,大家最容易想到的可能就是通过序列化/反序列化来攻击单例模式,因为一个对象实例序列化再反序列化后,得到的新的对象虽然各字段内容和原字段一致,然而对象地址和原始对象地址相比已经发生了变化,因此它们是两个不同的对象。 上面的结论完全正确,然而除了序列化/反序列化,单例模式还可能遭受另一种方式的攻击,即反射攻击(Reflection attack)。 看一个...
- 下一篇
这一堆初中生写的类库、框架,让一群中年程序员坐不住了!
前不久在V2EX上看到一个帖子,主题是「一堆初中生写的类库、框架」,原本猿妹以为只是个标题党不以为意。点进去该博主的博客主页发现,博主确实是一名初中生,而且他的GitHub主页,确实已经创建了多个项目,如下图: 虽然Star数量都不多,但是项目的技术水平却不低,下面就介绍几个项目给你们: 1、XPHP 根据他在V2EX上的介绍:这个项目,在他小学的时候就已经开发了第一个版本,那时还不知道 Semantic Version,于是版本号是版本0,这个框架旨在像 Laravel 一样优雅,与 CI 一样高性能。 它把整个 Handler 层单独出来了,所以整合 Swoole、Workerman 之类的十分方便。后期有原生支持多线程( pthreads )的打算。提供完整的工具集,包括 CLI 等等。Container 化管理,符合 PSR 标准,摒弃了 Laravel 中我认为多余的思想,十分简化,并且是 Event-Driven 的。 这个框架是完全模块化的,例如你觉得 Router 不好用,可以直接替换,只需修改 Container中的 Registry 就行了。(项目地址:https:...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Mario游戏-低调大师作品
- CentOS6,CentOS7官方镜像安装Oracle11G
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Windows10,CentOS7,CentOS8安装Nodejs环境
- Docker安装Oracle12C,快速搭建Oracle学习环境
- Windows10,CentOS7,CentOS8安装MongoDB4.0.16
- CentOS8编译安装MySQL8.0.19
- MySQL8.0.19开启GTID主从同步CentOS8
- CentOS8安装Docker,最新的服务器搭配容器使用
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7