原型模式(创建型)

原型模式

介绍

定义:用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型对象相同的新对象。

简单理解,就是当需要创建一个指定的对象时,我们刚好有一个这样的对象,但是又不能直接使用,我会clone一个一模一样的新对象来使用,这就是原型模式。关键字: Clone

原型模式分为”深拷贝”和”浅拷贝”。

深拷贝:创建一个新对象,对象的属性中引用的其他对象也会被克隆,不再指向原有对象地址;
浅拷贝:创建一个新对象,新对象的属性和原来对象完全相同,对于非基本类型属性(引用类型),仍指向原有属性所指向的对象的内存地址。

原型模式包含以下角色:

  • 抽象原型类: 规定了具体原型对象必须实现的 clone() 方法;
  • 具体原型类:实现抽象原型类的 clone() 方法,它是可被复制的对象;
  • 访问类:使用具体原型类中的 clone() 方法来复制新的对象;

Java中的 Object 类中提供了 clone() 方法来实现浅拷贝。 Cloneable 接口是抽象原型类,而实现了Cloneable接口的子实现类就是具体的原型类。

public class Prototype implements Cloneable {   // 具体原型类
    public Prototype() {
        System.out.println("具体的原型对象创建完成");
    }

    @Override
    protected Prototype clone() throws CloneNotSupportedException {
        System.out.println("具体原型复制成功!");
        return (Prototype) super.clone();
    }
}

public class PrototypeTest {    // 测试类
    public static void main(String[] args) throws CloneNotSupportedException {
        Prototype p1 = new Prototype();
        Prototype p2 = p1.clone();

        System.out.println("对象p1和p2是同一个对象?" + (p1 == p2));  // false
    }
}

再举个例子

现在正是秋招,投递简历会受到笔试的邮件,同一家公司发送的邮件内容除了名字不同,其他都相同,可以使用原型模式复制多个候选人邮件出来。

public class EmailPrototype implements Cloneable {
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public void show() {
        System.out.println(name + "同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!");
    }

    @Override
    protected EmailPrototype clone() throws CloneNotSupportedException {
        return (EmailPrototype) super.clone();
    }

    // 测试类
    static class EmailPrototypeTest {
        public static void main(String[] args) throws CloneNotSupportedException {
            EmailPrototype ep1 = new EmailPrototype();
            ep1.setName("张三");
            // 复制邮件
            EmailPrototype ep2 = ep1.clone();
            ep2.setName("李四");

            ep1.show();
            ep2.show();
        }
    }
}

// 运行结果:
张三同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!
李四同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!

注意点:如果具体原型类没有实现 Cloneable 接口,运行时会报异常。

使用场景:

  • 如果对象的创建非常复杂,可以使用原型模式快捷的创建对象;
  • 性能和安全要求比较高时;

(深浅拷贝)

将上面例子中属性 name 改成 Student 类型的属性,代码如下:

public class EmailPrototype2 implements Cloneable {
    // 学生类
    static class Student {
        private String name;
        private String university;

        // 省略 get set 方法

        public Student(String name, String university) {
            this.name = name;
            this.university = university;
        }
    }

    private Student stu;

    public EmailPrototype2(Student stu) {
        this.stu = stu;
    }

    public Student getStu() {
        return stu;
    }

    public void setStu(Student stu) {
        this.stu = stu;
    }

    public void show() {
        System.out.printf("来自%s的%s同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!%n", stu.getUniversity(), stu.getName());
    }

    @Override
    protected EmailPrototype2 clone() throws CloneNotSupportedException {
        return (EmailPrototype2) super.clone();
    }
    // 测试方法
    static class EmailPrototype2Test {
        public static void main(String[] args) throws CloneNotSupportedException {
            Student stu1 = new Student("张三", "北京大学");
            EmailPrototype2 e1 = new EmailPrototype2(stu1);

            EmailPrototype2 e2 = e1.clone();
            Student stu2 = e2.getStu();
            stu2.setName("李四");
            System.out.println("stu1和stu2是同一个对象?" + (stu2 == stu1));

            e1.show();
            e2.show();
        }
    }
}

// 运行结果
stu1和stu2是同一个对象?true
来自北京大学的李四同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!
来自北京大学的李四同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!

原型模式(创建型)

分析:e2引用对象是由e1拷贝来的,然后修改stu2中的name属性为李四,发现stu1也改了,说明stu1和stu2指向的是同一块堆内存(同一对象),这就是浅拷贝的效果。

但是这种场景下需要使用深拷贝,当前类和属性类都要实现 Cloneable 接口,代码如下:

public class EmailPrototype2 implements Cloneable {
    // 学生类
    static class Student implements Cloneable {
        private String name;
        private String university;

        // 省略 get set 方法

        public Student(String name, String university) {
            this.name = name;
            this.university = university;
        }

        @Override
        protected Student clone() throws CloneNotSupportedException {
            return (Student) super.clone();
        }
    }

    private Student stu;

    public EmailPrototype2(Student stu) {
        this.stu = stu;
    }

    public Student getStu() {
        return stu;
    }

    public void setStu(Student stu) {
        this.stu = stu;
    }

    void show() {
        System.out.printf("来自%s的%s同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!%n", stu.getUniversity(), stu.getName());
    }

    @Override
    protected EmailPrototype2 clone() throws CloneNotSupportedException {
        EmailPrototype2 ep = (EmailPrototype2) super.clone();
        ep.setStu(ep.getStu().clone());     // 实现深拷贝这句很关键
        return ep;
    }

    static class EmailPrototype2Test {
        public static void main(String[] args) throws CloneNotSupportedException {
            Student stu1 = new Student("张三", "北京大学");
            EmailPrototype2 e1 = new EmailPrototype2(stu1);

            EmailPrototype2 e2 = e1.clone();
            Student stu2 = e2.getStu();
            stu2.setName("李四");
            System.out.println("stu1和stu2是同一个对象?" + (stu2 == stu1));

            e1.show();
            e2.show();
        }
    }
}
// 运行结果:
stu1和stu2是同一个对象?false
来自北京大学的张三同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!
来自北京大学的李四同学:您好!非常高兴邀请您参加本公司2023校园招聘在线考试,希望您能按时完成!

原型模式(创建型)

改动的地方有以下几点:

  1. 属性类Student 也要实现 Cloneable 接口,并重写 clone() 方法;
  2. 邮件类 EmailPrototype2 中重写的 clone() 方法中,要对 Student 类型的属性进行克隆;

总结

原型模式的本质就是clone,可以解决构建复杂对象的资源消耗问题,能在某些场景中提升构建对象的效率

参考

(140条消息) 23 种设计模式详解(全23种)_鬼灭之刃的博客-CSDN博客_设计模式

https://www.bilibili.com/video/BV1Np4y1z7BU

Original: https://www.cnblogs.com/afei688/p/16730332.html
Author: 阿飞的客栈
Title: 原型模式(创建型)

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/644535/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • 谈谈微信支付曝出的漏洞

    来源: http://blog.w3pc.com/ 一、背景 昨天(2018-07-04)微信支付的SDK曝出重大漏洞(XXE漏洞),通过该漏洞,攻击者可以获取服务器中目录结构、文…

    技术杂谈 2023年5月31日
    01.3K
  • Dependencies与DepencyManagement的区别

    在maven的项目中经常遇到dependencies与dependencyManagement。那么两者有什么区别和联系呢? DepencyManagement的应用场景 主要应用…

    技术杂谈 2023年7月10日
    0100
  • 日常踩坑_jpa的踩坑心得

    背景提要 使用jpa的出现了很多问题1、使用between做日期范围查询时报错2、使用@Query注解写原生sql时报错3、使用@where注解自动在sql后添加条件时查不出东西4…

    技术杂谈 2023年7月25日
    098
  • 威胁分析和风险分析

    三个定义:威胁(Threat),指的是可能造成危害的来源。风险(Risk),可能造成的损失。漏洞,系统中可能被威胁利用从而造成危害的地方。 威胁分析就是把 系统中存在的威胁全部找出…

    技术杂谈 2023年6月21日
    0133
  • kfence源码分析

    参考 Linux mem 2.8 Kfence 详解 Linux 内核调测中最最让开发者头疼的 bug 有解了 让人头疼的”内核内存被改”和”内…

    技术杂谈 2023年5月30日
    0118
  • Accounting Calendar template

    SELECT INITCAP (TO_CHAR (TO_DATE (&year || ‘-‘ || LPAD (ROWNUM, 2, ‘…

    技术杂谈 2023年6月1日
    0134
  • DAX:跟关系相关的函数

    在表格数据模型中,用户可以创建关系,并可以沿着关系的方向自动进行交叉过滤。但是在 计算列中,必须通过RELATED 和 RELATEDTABLE函数来检索相关联的表。当使用CALC…

    技术杂谈 2023年5月31日
    0117
  • RDD(弹性分布式数据集)及常用算子

    RDD(弹性分布式数据集)及常用算子 RDD(Resilient Distributed Dataset)叫做弹性分布式数据集,是 Spark 中最基本的数据 处理模型。代码中是一…

    技术杂谈 2023年7月11日
    0112
  • C#中判断一个集合是另外一个集合的子集

    有这样的两个集合: string[] bigArr = new string[] { “a”, “b”, “c&#822…

    技术杂谈 2023年5月31日
    093
  • jboss_log4j.xml配置

    log4j是个优秀的开源的java日志系统,jboss内部也集成他,在jboss下默认的只是对server做了每日日志,并没有对你部署的项目进行每日的日志构建,但我们能通过修改jb…

    技术杂谈 2023年5月30日
    0102
  • java对形参操作能否改变实参

    这个问题其实以前就断断续续的纠结过,这次机缘巧合之下稍微深入的理解了这个问题。 这里的问题是:在主方法里创建了N个一般属性,将这些属性传递给其他方法,当 其他方法改变了传递来的形参…

    技术杂谈 2023年7月24日
    090
  • bare Git 仓库是什么?

    背景 今天,坐我旁边的同事问我一些关于服务器上命令的问题。其中有一个用了特殊参数的 git init 的命令,我也不认识,遂去 Google… bare Git 仓库 …

    技术杂谈 2023年7月10日
    093
  • 多级缓存-OpenResty获取请求参数

    OpenResty提供了各种API用来获取不同类型的请求参数: 在查询商品信息的请求中,通过路径占位符的方式,传递了商品id到后台: 需求:在OpenResty中接收这个请求,并获…

    技术杂谈 2023年5月31日
    0116
  • ATT&CK是什么

    一、ATT&CK官网 ATT&CK, Adversarial Tactics, Techniques, and Common Knowledge,对抗战术、技术与通…

    技术杂谈 2023年5月31日
    0191
  • 多项式求逆

    已知一度数为n的多项式(A(x))。 [A(x)B(x)\equiv1\pmod {x^n} ] (B(x))即为(A(x))的逆元。 多项式的除法、(\exp)和(\ln)都是基…

    技术杂谈 2023年6月21日
    0124
  • KL散度(距离)和JS散度(距离)zz

    两者都可以用来衡量两个概率分布之间的差异性。JS散度是KL散度的一种变体形式。 KL散度:也称相对熵、KL距离。对于两个概率分布P和Q之间的差异性(也可以简单理解成相似性),二者越…

    技术杂谈 2023年5月31日
    0121
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球