小王是学校的学霸,凭借着自己的天赋以及对于学习的刻苦,每次考试都能排到年级第一名。但是,他所在的班级总成绩却不高,在所有班级中,属于中游水平。老师希望通过小王的贡献,能否帮助整个班级同学分数都有一个提升。
老师跟小王提出了这个建议后,小王也很痛快就答应了。然后,利用周六和周日两天时间,将本周的一些重点和难点知识,以及他自己对于某些知识的理解小窍门都总结了出来,一共写了20多页的总结笔记。
周一上学,大家都争着抢着去借阅小王的“学习秘籍
”,甚至很多同学都开始手抄了起来。20多页的内容,纯手抄得需要好长时间,这时候,小王跟同学们说,“大家别手抄了,太麻烦了,我去楼下的打印社,给同学们每人打印一份”。
就这样,大概用了20分钟不到的时间,就给全班50多名同学每人打印了一份。大家开心的复习了起来,最后,通过小王同学每周总结的“学习秘籍”,他们班级的总成绩跃升成为了年级第一,大家开心极了!
通过上面的例子我们发现,如果大家手抄一份学习秘籍,假设每人平均需要1个小时的话,那么班级50名同学,都抄完就需要50个小时了。但是,如果复印的话,一页50份如果需要1分钟,那么20页需要20分钟就可以了。同样的创建50份学习秘籍,从50小时缩短为20分钟,这就是我们今天要介绍的设计模式的魅力——原型模式。
原型模式(Prototype Pattern
)
用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。也就是说,这种不通过new关键字来产生一个对象,而是通过对象复制(Java中的clone或反序列化)来实现的模式,就叫做原型模式。
原型模式的类图比较简单,只需要clone方法和实现clone方法即可,客户端如果需要去创建实例,则通过调用clone就可以了。具体类图如下所示:
性能优良
原型模式是在内存二进制流的拷贝,要比直接new一个对象性能好很多,特别是要在一个循环内产生大量的对象时,原型模式可能更好地体现其优点。
逃避构造函数的约束
直接在内存中拷贝,构造函数是不会执行的。
资源优化场景
类初始化需要消耗非常多的资源,这个资源包括数据、硬件资源等等。
性能和安全要求的场景
通过new产生一个对象需要非常繁琐的数据准备或访问权限,这时则可以使用原型模式。
一个对象多个修改者的场景
一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用。
验证通过clone()方法不会触发构造函数的执行:Prototype.java
- public class Prototype implements Cloneable {
- public static void main(String[] args) {
- Prototype prototype = new Prototype();
-
- /** 通过clone方法创建的Prototype对象不会执行构造方法 */
- Prototype clonePrototype = prototype.clone();
- }
-
- public Prototype() {
- System.out.println("-----Prototype的构造方法被执行了!-----");
- }
-
- @Override
- protected Prototype clone() {
- try {
- return (Prototype)super.clone();
- } catch (CloneNotSupportedException e) {
- e.printStackTrace();
- }
- return null;
- }
- }
执行结果如下所示:
- -----Prototype的构造方法被执行了!-----
-
- Process finished with exit code 0
实现浅拷贝与深拷贝:Prototype1.java
- @Data
- public class Prototype1 implements Cloneable {
-
- private String name;
-
- private List<String> arrayList = new ArrayList<>();
-
- public static void main(String[] args) {
- Prototype1 prototype1 = new Prototype1();
- prototype1.setName("orign object");
- prototype1.setValue("orign object");
-
- Prototype1 clonePrototype1 = prototype1.clone();
- clonePrototype1.setName("clone object");
- /** 发现添加了执行了clone对象的setValue之后,也修改了prototype1中的arrayList中数据 */
- clonePrototype1.setValue("clone object");
- System.out.println(prototype1);
- System.out.println(clonePrototype1);
- }
-
- /**
- * 浅拷贝
- * @return
- */
- @Override
- protected Prototype1 clone() {
- try {
- return (Prototype1)super.clone();
- } catch (CloneNotSupportedException e) {
- e.printStackTrace();
- }
- return null;
- }
-
- /**
- * 深拷贝
- * @return
- */
- // @Override
- // protected Prototype1 clone() {
- // Prototype1 prototype1 = null;
- // try {
- // prototype1 = (Prototype1)super.clone();
- // prototype1.setArrayList(new ArrayList<>());
- // } catch (CloneNotSupportedException e) {
- // e.printStackTrace();
- // }
- // return prototype1;
- // }
-
-
- public void setValue(String value) {
- this.arrayList.add(value);
- }
-
- public List<String> getValue() {
- return this.arrayList;
- }
- }
执行结果如下所示:
- Prototype1(name=orign object, arrayList=[orign object, clone object])
- Prototype1(name=clone object, arrayList=[orign object, clone object])
-
- Process finished with exit code 0
因为Object类提供的方法clone只是拷贝本对象,其对象内部的数组、引用对象等都不拷贝,还是指向原生对象的内部元素地址,这种拷贝就叫做浅拷贝。
今天的文章内容就这些了:
写作不易,笔者几个小时甚至数天完成的一篇文章,只愿换来您几秒钟的 点赞 & 分享 。
更多技术干货,欢迎大家关注公众号“爪哇缪斯” ~ \(^o^)/ ~ 「干货分享,每天更新」