当前位置: 首页 > news >正文

设计模式之原型模式(通俗易懂--代码辅助理解【Java版】)

文章目录

  • 设计模式概述
  • 1、原型模式
  • 2、原型模式的使用场景
  • 3、优点
  • 4、缺点
  • 5、主要角色
  • 6、代码示例
  • 7、总结
  • 题外话关于使用序列化实现深拷贝

设计模式概述

创建型模式:工厂方法、抽象方法、建造者、原型、单例。
结构型模式有:适配器、桥接、组合、装饰器、外观、享元、代理。
行为型模式有:责任链、命令、解释器、迭代器、中介、备忘录、观察者、状态、策略、模板方法、访问者。
常用设计模式:
单例模式、工厂模式、代理模式、策略模式&模板模式、门面模式、责任链模式、装饰器模式、组合模式、builder模式。

1、原型模式

  • 原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式之一。
  • 这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价比较大时,则采用这种模式。

2、原型模式的使用场景

  • 需要一个类的大量对象的时候,使用原型模式是最佳选择,因为原型模式是在内存中对这个对象进行拷贝,要比直接new这个对象性能要好很多,在这种情况下,需要的对象越多,原型模式体现出的优点越明显。
  • 如果一个对象的初始化需要很多其他对象的数据准备或其他资源的繁琐计算,那么可以使用原型模式。
  • 当需要一个对象的大量公共信息,少量字段进行个性化设置的时候,也可以使用原型模式拷贝出现有对象的副本进行加工处理。

3、优点

  • 原型模式允许在运行时动态改变具体的实现类型。原型模式可以在运行期间,由客户来注册符合原型接口的实现类型,也可以动态地改变具体的实现类型,看起来接口没有任何变化,但其实运行的已经是另外一个类实例了。因为克隆一个原型就类似于实例化一个类。

4、缺点

  • 原型模式最主要的缺点是每一个类都必须配备一个克隆方法。配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类来说不是很难,而对于已经有的类不一定很容易,特别是当一个类引用不支持序列化的间接对象,或者引用含有循环结构的时候

5、主要角色

(1)客户(Client)角色:客户类提出创建对象的请求。
(2)抽象原型(Prototype)角色:这是一个抽象角色,通常由一个Java接口或Java抽象类实现。此角色给出所有的具体原型类所需的接口。
(3)具体原型(Concrete Prototype)角色:被复制的对象。此角色需要实现抽象的原型角色所要求的接口。

6、代码示例

1)UML图
在这里插入图片描述
2)源代码
(1)抽象原型角色

public interface Prototype{/*** 克隆自身的方法* @return 一个从自身克隆出来的对象*/public Object clone();
}

(2)具体原型角色


public class ConcretePrototype1 implements Prototype {public Prototype clone(){//最简单的克隆,新建一个自身对象,由于没有属性就不再复制值了Prototype prototype = new ConcretePrototype1();return prototype;}
}public class ConcretePrototype2 implements Prototype {public Prototype clone(){//最简单的克隆,新建一个自身对象,由于没有属性就不再复制值了Prototype prototype = new ConcretePrototype2();return prototype;}
}

(3)客户端角色

public class Client {public static void main(String[]args){try{Prototype p1 = new ConcretePrototype1();//获取原型来创建对象Prototype p2 = p1.clone();//有人动态的切换了实现Prototype p3 = new ConcretePrototype2();Prototype p4 = p3.clone();}catch(Exception e){e.printStackTrace();}}
}
}

7、总结

  • 这种情况就是浅拷贝,java只拷贝你指定的对象,至于你指定的对象里面的别的对象,它不拷贝,还是把引用给你,共享变量,这是一种非常不安全的方式,需要特别注意。
  • 内部的数组和引用对象不会拷贝,其他的原始基本类型和String类型会被拷贝。
  • 如果需要可以使用深拷贝,结合具体对象的情况进行处理(可以自己实现深拷贝逻辑;或者利用序列化和反序列化实现深拷贝,前提是所有需要复制的对象都需要实现java.io.Serializable接口。

题外话关于使用序列化实现深拷贝

利用序列化实现深度克隆:

  • 把对象写到流里的过程是序列化(Serialization)过程;而把对象从流中读出来的过程则叫反序列化(Deserialization)过程。应当指出的是,写到流里的是对象的一个拷贝,而原对象仍然存在于JVM里面。
  • 在Java语言里深度克隆一个对象,常常可以先使对象实现Serializable接口,然后把对象(实际上只是对象的拷贝)写到一个流里(序列化),再从流里读回来(反序列化),便可以重建对象。
public  Object deepClone() throws IOException, ClassNotFoundException{//将对象写到流里ByteArrayOutputStream bos = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(bos);oos.writeObject(this);//从流里读回来ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());ObjectInputStream ois = new ObjectInputStream(bis);return ois.readObject();}
  • 这样做的前提就是对象以及对象内部所有引用到的对象都是可序列化的,否则,就需要仔细考察那些不可序列化的对象可否设成transient,从而将之排除在复制过程之外。

  • 浅度克隆显然比深度克隆更容易实现,因为Java语言的所有类都会继承一个clone()方法,而这个clone()方法所做的正式浅度克隆。

http://www.lryc.cn/news/453326.html

相关文章:

  • Study-Oracle-10-ORALCE19C-RAC集群维护
  • 【无题】夜入伊人笑愉,泪湿心夜难眠。
  • docker下载mysql时出现Unable to pull mysql:latest (HTTP code 500) server error 问题
  • 厦门网站设计的用户体验优化策略
  • Fastjson反序列化
  • Python Linux解压安装脚本
  • numpy 逻辑运算方法介绍
  • 怎么查看网站是否被谷歌收录,查看网站是否被谷歌收录的简便方法
  • 【前端开发入门】前端开发环境配置
  • Windows驱动开发(二)
  • Hotspot是什么?
  • k8s-集群部署1
  • wordpress Contact form 7发件人邮箱设置
  • 15分钟学 Python 第38天 :Python 爬虫入门(四)
  • GWAS分析中显著位点如何注释基因:excel???
  • 深入浅出 CSS 定位:全面解析与实战指南
  • HTTPS协议详解:从原理到流程,全面解析安全传输的奥秘
  • Android 13.0 系统内存优化之修改dalvik虚拟机的内存参数
  • C# 无边框窗体,加阴影效果、多组件拖动、改变大小等功能完美实现优化版效果体验
  • 深入解析 ChatGLM 模型:核心原理、优势与未来应用前景
  • python全栈学习记录(二十二)多态性、封装、绑定方法与非绑定方法
  • 用Python制作自己的聊天机器人:从零开始构建智能对话助手
  • LabVIEW裂纹深度在线监测系统
  • 工业物联网的伦理和社会影响
  • TCP --- 确认应答机制以及三次握手四次挥手
  • GPT带我学-设计模式17-装饰器模式
  • 【Redis】如何在 Ubuntu 上安装 Redis 5
  • 房屋水电费记账本:内置的数组数据击按钮不能删除,页面手动添加的可以删除
  • 【ubuntu】apt是什么
  • 堆排序算法的原理与应用