原型模式:啥?盗图、盗文章的人居然用的是一种设计模式!

2023-12-25 11:38

本文主要是介绍原型模式:啥?盗图、盗文章的人居然用的是一种设计模式!,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

clone-wordpress

原型模式 🎭

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

来源:https://refactoringguru.cn/design-patterns/prototype

图片来源:https://refactoringguru.cn/design-patterns/prototype

月饼?盗文章? 🤔

每年中秋节的时候,大家都会吃到自己心仪口味的样式各异的月饼,但是他是怎么生产出来的呢,我猜它应该是有一个模板,比如花边图案的月饼

月饼模板原型

他会创造出来一个月饼原型,当你想吃五仁的时候,就把里面的馅改成五仁的,当你想吃蛋黄的(自己准备鸡蛋),就把馅改成蛋黄的,这样做不仅提高了生产效率,而且还节省了一部分再创建一个月饼的时间。

再比如说可恶的盗图、盗文章、盗视频的人,他们把原创内容拿回去改个名字,去掉水印,随便改改内容,就成了自己的了!?

clone-wordpress1

上面说的两个案例的行为都是在节省了创建时间,同时达到了自己的一些目的。而这在设计模式中,就叫做原型模式,为了解决一个对象的创建而出现的一种设计模式,归类在了创建型模式中。

注意:在行为型模式中有着与之类似的一种模式——模版方法模式,是为了制定解决一件事情中的一系列操作约束而存在的一种模式,二者的区别在于,原型模式用于对象创建,模板方法模式用于事件行为约束。

原型模式类图 📌

prototype-UML

来看代码 📃

完整代码及单元测试结果点击查看 https://github.com/lvgocc/java-design-patterns/tree/main/prototype

java 的 JDK 中,提供一个标识接口 Cloneable ,我们将需要定义成原型的类实现这个接口就可以完成复制了。

通过 cloneable 接口实现原型模式

public class Graphics implements Cloneable {private final String color;private final String shape;public Graphics(String color, String shape) {this.color = color;this.shape = shape;}@Overrideprotected Graphics clone() throws CloneNotSupportedException {return (Graphics) super.clone();}..... set/get/toString
}

在上面的例子中,类中的对象类型都是基本类型,如果出现引用类型的时候,就会引发一个问题“浅克隆”,这会导致我们克隆出来的类会受原型中引用的类型影响,那我们如何才能规避这个问题做到“深克隆”呢​?

浅克隆?🎭 深克隆?🎎

浅、深指的是对对象的占有权利。比如我借给你一个手机,那你只能使用这个手机里现有的东西,我如果删除了一个软件,那你自然也就没有这个软件了。假如我送给你一个手机,那你就可以随便的使用,不用担心我会做什么了,因为这个手机就是你的了。

那在 java 代码中,我们怎么理解浅克隆🎭 、深克隆🎎 呢?

实现了 cloneable 接口,可以克隆一个区别于当前对象的另外一个新的对象,但对于对象中的引用,却不能进行克隆,你虽得到了他的人,但你却得不到他的心,如果想要得到他的心怎么办?拿钱砸他!非也,你只要把他的心也克隆一份就可以了。但是前提是他的心允许克隆(实现了 cloneable 接口)。

浅克隆代码

public class Graphics implements Cloneable {private final String color;private final String shape;// 引用类型没有实现 cloneable 接口private final Size size;public Graphics(String color, String shape, Size size) {this.color = color;this.shape = shape;this.size = size;}@Overrideprotected Graphics clone() throws CloneNotSupportedException {return (Graphics) super.clone();}
}

引用类型没有实现 cloneable 接口

// 引用类型没有实现 cloneable 接口
public class Size {public int width;public int height;public Size(int width, int height) {this.width = width;this.height = height;}@Overridepublic String toString() {return "Size(" + width + ", " + height + ")";}
}

测试结果

class GraphicsTest {@Testvoid graphicsTest() throws CloneNotSupportedException {Size size = new Size(1, 2);Graphics graphics = new Graphics("red", "circular", size);Graphics clone = graphics.clone();size.height = 3;size.width = 5;System.out.println("graphics = " + graphics);// 判断两个对象是否不同Assertions.assertNotSame(graphics, clone);clone.setColor("blue");clone.setShape("square");System.out.println("clone = " + clone);}
}

注意此时的引用对象 size 的值
修改引用类型内容导致 clone 类的内容也跟着发生了变化

graphics = Graphics[color='red', shape='circular', size=Size(5, 3)]
clone = Graphics[color='blue', shape='square', size=Size(5, 3)]

深克隆:引用类型也实现 Cloneable 接口

// 引用类型实现了 cloneable 接口
public class Size implements Cloneable {public int width;public int height;public Size(int width, int height) {this.width = width;this.height = height;}@Overrideprotected Size clone() throws CloneNotSupportedException {return (Size) super.clone();}@Overridepublic String toString() {return "Size(" + width + ", " + height + ")";}
}

在原型类中调整 clone 方法

    @Overrideprotected Graphics clone() throws CloneNotSupportedException {Graphics clone = (Graphics) super.clone();clone.size = size.clone();return clone;}

测试结果

class GraphicsTest {@Testvoid graphicsTest() throws CloneNotSupportedException {Size size = new Size(1, 2);Graphics graphics = new Graphics("red", "circular", size);Graphics clone = graphics.clone();// 修改引用类型内容size.height = 3;size.width = 5;System.out.println("graphics = " + graphics);// 判断两个对象是否不同Assertions.assertNotSame(graphics, clone);clone.setColor("blue");clone.setShape("square");System.out.println("clone = " + clone);}
}

注意此时的引用对象 size 的值

graphics = Graphics[color='red', shape='circular', size=Size(5, 3)]
clone = Graphics[color='blue', shape='square', size=Size(1, 2)]

完整代码 https://github.com/lvgocc/java-design-patterns/tree/main/prototype

原型模式自身有什么优势和问题呢?

优势:

  1. JDK 的 cloneable 接口是基于内存数据的直接复制,速度相较于 new 关键字创建对象更加快速;同时简化了创建过程(不会执行构造方法)。
  2. 通过深克隆来保存一个对象某一时刻的状态,便于还原,实现撤销操作;

问题:

  1. 需要为每个类重写 #clone 方法;
  2. 深克隆需要将每个对象都维护一个 cloneable 接口;
  3. 构造方法中的代码不会执行;

总结

当我们需要频繁使用一些类似的对象的时候,可以考虑使用原型模式来降低资源的开销,使资源得到合理的分配和使用。而对于原型模式的深克隆带来的弊端,就显得那么的不重要了。

  1. 类似的对象使用频繁,考虑原型模式
  2. 深克隆时注意类中的引用类型是否实现了 cloneable 接口
  3. 注意构造函数中是否有必要代码要执行,可以考虑放到 #clone 方法中执行

写在最后

Java 设计模式专题,共23 种设计模式。内容来自个人学习理解消化的结果,谈不上教程,只望记录于此同你分享。希望能够和大家一起进步、成长。为了梦想,学习技术。如果你觉得文章对你有帮助,希望随便点赞、关注、分享、给个 star 支持一下。感激涕零🎈。

⭐https://github.com/lvgocc/java-design-patterns

星尘的一个朋友 亦或繁星、亦或尘埃。星尘,为了梦想,学习技术。

关注公众号,回复“加群”,和 lvgo 一起学习设计模式。坚决维护群内环境,给大家提供一个良好的学习交流环境。

星尘的一个朋友

这篇关于原型模式:啥?盗图、盗文章的人居然用的是一种设计模式!的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/535383

相关文章

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

模版方法模式template method

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/template-method 超类中定义了一个算法的框架, 允许子类在不修改结构的情况下重写算法的特定步骤。 上层接口有默认实现的方法和子类需要自己实现的方法

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

迭代器模式iterator

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/iterator 不暴露集合底层表现形式 (列表、 栈和树等) 的情况下遍历集合中所有的元素

《x86汇编语言:从实模式到保护模式》视频来了

《x86汇编语言:从实模式到保护模式》视频来了 很多朋友留言,说我的专栏《x86汇编语言:从实模式到保护模式》写得很详细,还有的朋友希望我能写得更细,最好是覆盖全书的所有章节。 毕竟我不是作者,只有作者的解读才是最权威的。 当初我学习这本书的时候,只能靠自己摸索,网上搜不到什么好资源。 如果你正在学这本书或者汇编语言,那你有福气了。 本书作者李忠老师,以此书为蓝本,录制了全套视频。 试

利用命令模式构建高效的手游后端架构

在现代手游开发中,后端架构的设计对于支持高并发、快速迭代和复杂游戏逻辑至关重要。命令模式作为一种行为设计模式,可以有效地解耦请求的发起者与接收者,提升系统的可维护性和扩展性。本文将深入探讨如何利用命令模式构建一个强大且灵活的手游后端架构。 1. 命令模式的概念与优势 命令模式通过将请求封装为对象,使得请求的发起者和接收者之间的耦合度降低。这种模式的主要优势包括: 解耦请求发起者与处理者

springboot实战学习(1)(开发模式与环境)

目录 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 (3)前端 二、开发模式 一、实战学习的引言 (1)前后端的大致学习模块 (2)后端 Validation:做参数校验Mybatis:做数据库的操作Redis:做缓存Junit:单元测试项目部署:springboot项目部署相关的知识 (3)前端 Vite:Vue项目的脚手架Router:路由Pina:状态管理Eleme

状态模式state

学习笔记,原文链接 https://refactoringguru.cn/design-patterns/state 在一个对象的内部状态变化时改变其行为, 使其看上去就像改变了自身所属的类一样。 在状态模式中,player.getState()获取的是player的当前状态,通常是一个实现了状态接口的对象。 onPlay()是状态模式中定义的一个方法,不同状态下(例如“正在播放”、“暂停

软件架构模式:5 分钟阅读

原文: https://orkhanscience.medium.com/software-architecture-patterns-5-mins-read-e9e3c8eb47d2 软件架构模式:5 分钟阅读 当有人潜入软件工程世界时,有一天他需要学习软件架构模式的基础知识。当我刚接触编码时,我不知道从哪里获得简要介绍现有架构模式的资源,这样它就不会太详细和混乱,而是非常抽象和易