首页 > 其他分享 >原型模式

原型模式

时间:2022-12-26 10:37:08浏览次数:32  
标签:Graphic 对象 clone 模式 实例 乐符 原型

概述

《设计模式》一书中对 “原型模式” 的意图描述如下:

用原型实例指定创建对象的种类,并通过拷贝对象的原型创建新的对象

貌似有点难懂,概括性地讲就是实现对不同类型对象的复制,而不需要知道他们的具体类型

一般情况下的类结构图如下图所示:

Prototype.png

原型模式适用于以下场景:

  • 当一个系统需要独立于它的产品创建、构成和表示时
  • 当要实例化的类是在运行时指定
  • 为了避免创建一个与产品层次平行的工厂类层次时
  • 当一个类的实例只能有几个不同状态组合中的一种时

具体实例

以 《设计模式》一书中提到的 “动机” 为例:现在需要创建一个通用的图形编辑器框架,以及增加一些表示音符、休止符和五线谱的新对象来构造一个乐谱编辑器。这个编辑器框架可能有一个工具选择板用于将这些音乐对象加入到乐谱中,同时,这个选择板可能还包括选择、移动和其它操纵音乐对象的工具。

首先我们定义所有图形对象都具备的接口 Graphic

public interface Graphic {
    // 图形对象具备的公共行为
}

然后是针对乐符对象的操作类的接口 GraphicTool

public interface GraphicTool {
    void moveToBashboard(Graphic g); // 移动乐符对象到工具版
}

现在,需要考虑如何移动乐符对象。如果乐符在系统中每种类型都是唯一的,那么直接将这些乐符对象设置为单例对象进行移动即可。然而,系统中可能存在不同类型的乐符,这些乐符之间的属性可能并不一样,甚至在移动乐符之后这些属性都会发生变化,因此设置为单例的方案是不可行的。

为了确保这些乐符之间不会相互影响,我们可以在移动乐符的时候创建对应的副本实例,从而隔离乐符对象之间的相互影响。然而,对于 Graphic 对象来讲,我们并不知道具体的类型,因此无法对乐符对象进行对应的实例化

可行的解决方案是在 Graphic 中定义一个方法,要求具体的子类实例类能够自己创建对应的副本对象,那么 GraphicTool 在移动乐符对象时就不需要再考虑对象实例化的问题,同时,对于可以设置为单例对象的乐符,也可以由子类自定义进行浅拷贝

最终,我们的 Graphic 会存在一个公共的 clone 方法以返回对应的副本实例:

public interface Graphic {
    void draw(Position p); // 假设存在方法将图形绘画到指定位置
    
    Graphic clone(); // 创建自身副本对象
}

而我们的 GraphicTool 在移动乐符对象时,可以自行获取对应的乐符对象进行操作:

public abstract class AbstractGraphicTool 
    implements GraphicTool {
    @Override
    public void moveToBashboard(Graphic g) {
        Graphic p = g.clone();
        p.draw(new Position());
    }
}

总结

尽管原型模式存在许多的优点,但是实际用起来不是那么方便,因为没有确切的手段能够保证对象的克隆一定是深拷贝。因此在实现时,对于 clone() 方法的实现难度很大,一般情况下也不建议直接使用原型模式。特别是对于 Java 中 Object 定义的 clone() 方法,一般都不推荐重写它。一种比较合适的方式是对于 clone() 方法的实现通过重新构造对象实例来代替默认的 clone(),具体可以查看 《Effective Java(第三版)》中第13条相关的描述


参考:

[1] 《设计模式—可复用面向对象基础》

标签:Graphic,对象,clone,模式,实例,乐符,原型
From: https://www.cnblogs.com/FatalFlower/p/17005142.html

相关文章

  • UML设计模式符号答疑
    在UML类图中,常见的有以下几种关系:泛化(Generalization),  实现(Realization),关联(Association),聚合(Aggregation),组合(Composition),依赖(Dependency) 1.泛化(Generalization)......
  • MongoDB - 模式设计
    注意事项模式设计,即在文档中表示数据的方式,对于数据表示来说时非常关键的。为MongoDB做模式设计时,在性能、可伸缩性和简单性方面是重中之重,也需要考虑一些特别的注意事......
  • 分布式事务方案 - SAGA模式
    本文目的是讲清楚SAGA这种分布式事务解决方案的实现思路,不包括具体实现代码,具体实现推荐使用阿里的Seata框架。内容包括:分布式事务问题描述SAGA-Choreography策......
  • 精通visual c++指纹模式识别系统算法及实现
    通过学习,掌握以下几个问题:1、核心算法,并且向GVF衍生;2、核心库封装的方法2016年11月16日06:52:51昨日实现了梯度场和频率场的计算。最大的感觉就是建立基础代码库的重要性。......
  • 【模式识别】判别函数和感知机算法
    给定两个类别的一共8个样本,通过感知机算法求解判别函数defbi_perception():X1=np.array([[0,0,0],[1,0,0],[1,0,1],......
  • CPU的实模式
    实模式是指CPU的寻址方式,寄存器大小,指令用法等,是用来反应CPU在该环境下如何工作的概念。CPU的工作原理:CPU大体分为三个部分,控制、运算、存储单元。控制单元是CPU的控制中......
  • 1个月连载30个设计模式真实案例,挑战年薪60W不是梦
    本文所有内容均节选自《设计模式就该这样学》本文自2021年10月29日起持续连载,请大家持续关注....序言DesignPatterns:ElementsofReusableObject-OrientedSoft......
  • 设计模式面试题(总结最全面的面试题!!!)
    Java面试总结汇总,整理了包括Java重点知识,以及常用开源框架,欢迎大家阅读。文章可能有错误的地方,因为个人知识有限,欢迎各位大佬指出!文章持续更新中......ID标题地址......
  • 第一百一十七篇: JavaScript 工厂模式和原型模式
    好家伙,本篇为《JS高级程序设计》第八章“对象、类与面向对象编程”学习笔记 1.工厂模式工厂模式是另外一种关注对象创建概念的创建模式。它的领域中同其它模式的不同......
  • 分组加密——多组加密模式
    分组加密——多组加密模式电子密码本EBCECB模式是将明文分成固定长度的组块,然后对每个明文分组块使用相同的密钥进行独立加密或解密,且该加密的结果将直接作为密文分组。......