趣解设计模式之《小王的学习秘籍》
  m5xpDbUW5OKT 2023年11月19日 23 0

〇、小故事

小王是学校的学霸,凭借着自己的天赋以及对于学习的刻苦,每次考试都能排到年级第一名。但是,他所在的班级总成绩却不高,在所有班级中,属于中游水平。老师希望通过小王的贡献,能否帮助整个班级同学分数都有一个提升

趣解设计模式之《小王的学习秘籍》_java

老师跟小王提出了这个建议后,小王也很痛快就答应了。然后,利用周六和周日两天时间,将本周的一些重点和难点知识,以及他自己对于某些知识的理解小窍门都总结了出来,一共写了20多页的总结笔记

趣解设计模式之《小王的学习秘籍》_后端_02

周一上学,大家都争着抢着去借阅小王的“学习秘籍”,甚至很多同学都开始手抄了起来。20多页的内容,纯手抄得需要好长时间,这时候,小王跟同学们说,“大家别手抄了,太麻烦了,我去楼下的打印社,给同学们每人打印一份”。

趣解设计模式之《小王的学习秘籍》_构造函数_03

就这样,大概用了20分钟不到的时间,就给全班50多名同学每人打印了一份。大家开心的复习了起来,最后,通过小王同学每周总结的“学习秘籍”,他们班级的总成绩跃升成为了年级第一,大家开心极了!

通过上面的例子我们发现,如果大家手抄一份学习秘籍,假设每人平均需要1个小时的话,那么班级50名同学,都抄完就需要50个小时了。但是,如果复印的话,一页50份如果需要1分钟,那么20页需要20分钟就可以了。同样的创建50份学习秘籍,从50小时缩短为20分钟,这就是我们今天要介绍的设计模式的魅力——原型模式

一、模式定义

原型模式Prototype Pattern

用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。也就是说,这种不通过new关键字来产生一个对象,而是通过对象复制(Java中的clone反序列化)来实现的模式,就叫做原型模式。

二、模式类图

原型模式的类图比较简单,只需要clone方法和实现clone方法即可,客户端如果需要去创建实例,则通过调用clone就可以了。具体类图如下所示:

趣解设计模式之《小王的学习秘籍》_原型模式_04

三、原型模式的应用场景

3.1> 原型模式的特点

性能优良

原型模式是在内存二进制流的拷贝,要比直接new一个对象性能好很多,特别是要在一个循环内产生大量的对象时,原型模式可能更好地体现其优点。

逃避构造函数的约束

直接在内存中拷贝,构造函数是不会执行的。

3.2> 原型模式的使用场景

资源优化场景

类初始化需要消耗非常多的资源,这个资源包括数据、硬件资源等等。

性能和安全要求的场景

通过new产生一个对象需要非常繁琐的数据准备或访问权限,这时则可以使用原型模式。

一个对象多个修改者的场景

一个对象需要提供给其他对象访问,而且各个调用者可能都需要修改其值时,可以考虑使用原型模式拷贝多个对象供调用者使用

四、原型模式的注意事项

4.1> 构造函数不会被执行

验证通过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

4.2> 浅拷贝&深拷贝

实现浅拷贝与深拷贝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^)/ ~ 「干货分享,每天更新」

【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月19日 0

暂无评论

推荐阅读
m5xpDbUW5OKT