首页 > 其他分享 >初识设计模式 - 原型模式

初识设计模式 - 原型模式

时间:2022-09-25 13:44:04浏览次数:51  
标签:对象 DeepProtoType 数据类型 初识 原型 拷贝 设计模式 public

简介

对于大部分系统来说,创建对象包括申请内存、给成员变量赋值等过程,这些操作耗费的时间基本可以忽略不计。

如果对象中的数据需要经过复杂的计算才能得到(比如排序、计算哈希值),或者需要从 RPC、网络、数据库、文件系统等非常慢速的 IO 中读取,这其中耗费的时间有时是无法容忍的。

如果对象的创建成本比较大,而同一个类的不同对象之间差别不大(如大部分字段都相同),在这种情况下,可以利用对已有对象(原型)进行复制的方式来创建新对象,以达到节省创建时间的目的。

这种基于原型来创建对象的方式就叫作原型设计模式(Prototype Design Pattern)。

具体实现

原型模式其核心就是拷贝,而在编程语言中,拷贝有两种实现方式:深拷贝和浅拷贝。

浅拷贝

浅拷贝图解

浅拷贝只会复制对象中基本数据类型和引用对象的内存地址,不会递归地复制引用对象,以及引用对象的引用对象……

对于浅拷贝来说,如果要拷贝的对象是不可变对象,浅拷贝共享不可变对象是没问题的,但是对于可变对象来说,浅拷贝得到的对象和原始对象会共享部分数据,就有可能出现数据被修改的风险。

深拷贝

image

和浅拷贝不同,深拷贝得到的是一份完完全全独立的对象,相比浅拷贝,深拷贝更加耗时、更加耗内存空间。

常见的深拷贝有两种实现方式:递归浅拷贝对象直到只包含基本数据类型数据;序列化对象再反序列化成新的对象。实际开发中更推荐使用后者。

递归浅拷贝的 Java 代码示例如下:

public class DeepProtoType implements Cloneable {
    // String 类型
    public String name;
    // 引用数据类型
    public DeepCloneableTarget deepCloneableTarget;

    public DeepProtoType() {
        super();
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        Object deep = null;
        // 这里完成对基本数据类型(属性)和 String 类型的拷贝
        deep = super.clone();
        // 对引用数据类型的属性进行单独处理,如果有多属性,需要一一处理
        DeepProtoType deepProtoType = (DeepProtoType)deep;
        deepProtoType.deepCloneableTarget = (DeepCloneableTarget)deepCloneableTarget.clone();

        return deepProtoType;
    }
}

序列化对象的代码示例如下:

public class DeepProtoType implements Serializable {
    // String 类型
    public String name;
    // 引用数据类型
    public DeepCloneableTarget deepCloneableTarget;

    public DeepProtoType() {
        super();
    }

    public Object deepClone() {
        // 创建流对象
        ByteArrayOutputStream bos = null;
        ObjectOutputStream oos = null;
        ByteArrayInputStream bis = null;
        ObjectInputStream ois = null;

        try {
            // 序列化
            bos = new ByteArrayOutputStream();
            oos = new ObjectOutputStream(bos);
            // 当前这个对象以对象流的方式输出
            oos.writeObject(this);

            // 反序列化
            bis = new ByteArrayInputStream(bos.toByteArray());
            ois = new ObjectInputStream(bis);
            DeepProtoType deepProtoType = (DeepProtoType) ois.writeObject();

            return deepProtoType;
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        } finally {
            try {
                bos.close();
                oos.close();
                bis.close();
                ois.close();
            } catch (Exception e2) {
                System.out.println(e2.getMessage());
            }
        }
    }
}

总结

优点

原型模式的主要优点如下:

  • 当创建新的对象实例较为复杂时,使用原型模式可以简化对象的创建过程,通过复制一个已有实例可以提高新实例的创建效率
  • 可以使用深拷贝方式保存对象的状态,使用原型模式将对象复制一份,并将其状态保存起来
  • 原型模式提供了简化的创建结构

缺点

原型模式的主要缺点如下:

  • 需要为每一个类都配置一个拷贝方法
  • 拷贝方法位于类的内部,当对已有类进行改造的时候,需要修改代码,违背了开闭原则
  • 当实现深拷贝的时候,需要编写较为复杂的代码,而且当对象之间存在多重嵌套引用时,为了实现深拷贝,每一层对象对应的类都必须支持深拷贝

适用场景

原型模式的适用场景如下:

  • 创建对象的成本较大,比如初始化时间长,占用 CPU 太多,或者占用网络资源太多等
  • 如果系统要保存对象的状态,而对象的状态变化很小,或者对象本身占用内存较小时,可以使用原型模式配合备忘录模式来实现
  • 如果产生一个对象需要非常繁琐的数据准备和访问权限,需要提高权限或提高安全性

源码

在 JDK 中,对自定义的类重写 clone() 方法就是典型的原型模式。

标签:对象,DeepProtoType,数据类型,初识,原型,拷贝,设计模式,public
From: https://www.cnblogs.com/fatedeity/p/16727727.html

相关文章

  • 06. 不急于开始原型设计;我知道你很兴奋!
    06.不急于开始原型设计;我知道你很兴奋!设计思维不仅仅适用于用户体验设计师。我很感激我的开发者们单枪匹马地开发了一个产品;你在家庭中扮演父母双方的角色。你必须做出......
  • Steamlit初识和安装入门
    一、前言之前看我同事搭建了一个简易的网页,觉得挺有趣的,一直想学习下来着,然而机缘巧合,前阵子才知道原来是用Streamlit开发的呀,于是又开始了自己的三分钟热情了,哈哈哈......
  • 微服务架构设计模式
    模块化是开发大型、复杂应用程序的基础微服务架构可以消除对某项技术栈的长期依赖微服务架构的弊端和问题:服务的拆分和定义是一项挑战;分布式系统带来的各种复杂性,使开......
  • 简明设计模式---桥接模式
    简述类型:结构型目的:通过抽离出多个维度相互组合(聚合)来代替继承,简化系统。话不多说,看个优化案例。优化案例现有系统中,对于画面窗口的边框有一套样式来控制是否有圆......
  • java设计模式之七大原则
    java设计模式 以下内容为本人的学习笔记,如需要转载,请声明原文链接   https://www.cnblogs.com/lyh1024/p/16724932.html 设计模式1.设计模式的目的编写软件过......
  • JavaScript prototype(原型对象)
     对象都会从一个prototype(原型对象)中继承属性和方法。添加属性和方法:想要在对象的构造函数中添加属性或方法使用prototype属性就可以给对象的构造函数添加新的属性......
  • 初识html1
    一.概念web1.网站:网页的集合(使用HTML等工具制作)2.网页:构成网站的基本因素(图片、文字、声音、视频、链接)3.HTML:超文本标记语言,HTML不是编程语言    超文本:(1)可以......
  • 初识unity1 02
    打组hierarchy面板中将一个物体拖拽到另一个物体中作为子物体(创建一个空物体作为父物体)子物体有了父亲以后,position变为与其父物体的相对距离(可以先直接对子物体reset)(子......
  • 设计模式_装饰模式
    先看个例子:你买了一辆爱车,现在想给车子增加装饰? 定义汽车行为接口【IBaoMa】,定义方法【startEngine】,【switchOnLights】,【start】。   定义爱车类【LoveCars】,实......
  • 【设计模式】之策略模式
    场景引入假如你在A城市,要去B城市旅游,交通方式有以下几种选择:驾车火车飞机不难写出这样的代码:voidtransport(stringmethod){if(method=="drive"){......