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

三、原型模式

一、什么是原型模式

  原型(Prototype)模式的定义如下:用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型相同或相似的新对象。在这里,原型实例指定了要创建的对象的种类。用这种方式创建对象非常高效,根本无须知道对象创建的细节。原型模式包含以下主要角色。

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

二、原型模式的实现

  深克隆与与浅克隆:Object类的clone方法只会克隆对象中的基本的数据类型,对于数组、容器对象、引用对象等都不会克隆,这就是浅克隆。如果要实现深克隆,必须将原型模式中的数组、容器对象、引用对象等另行克隆。
  由于 Java 提供了对象的 clone() 方法,所以用 Java 实现原型模式很简单。
1、创建Prototype的成员属性:


/*** @author FluffyCatkin* @version 1.0* @date 2021-03-31 0:12* @description Prototype的成员变量*/
public class Member implements Cloneable{private String position;public Member(String position) {this.position = position;}public void setPosition(String position) {this.position = position;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}@Overridepublic String toString() {return "Member{" +"position='" + position + '\'' +'}';}
}

2、创建具体原型类Prototype:

import java.util.ArrayList;public class Prototype implements Cloneable{private String name ;private int age;private ArrayList<String> hobbies;private Member member;public Prototype(String name, int age, ArrayList<String> hobbies, Member member) {this.name = name;this.age = age;this.hobbies = hobbies;this.member = member;System.out.println("通过构造方法创建对象。。。。。。");}/*** 浅克隆*/@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}/*** 深克隆*/
//    @Override
//    protected Object clone() throws CloneNotSupportedException {
//        Prototype clone = (Prototype) super.clone();
//        clone.setHobbies((ArrayList<String>) clone.getHobbies().clone());
//        clone.setMember((Member) clone.getMember().clone());
//        return clone;
//    }public void setName(String name){this.name = name;}public void setHobbies(ArrayList<String> hobbies) {this.hobbies = hobbies;}public ArrayList<String> getHobbies() {return hobbies;}public Member getMember() {return member;}public void setMember(Member member) {this.member = member;}@Overridepublic String toString() {return "Prototype{" +"name='" + name + '\'' +", age=" + age +", hobbies=" + hobbies +", member=" + member +'}';}
}

3、创建测试类(访问类):


import java.util.ArrayList;/*** 原型模式*  原型模式的定义与特点:*      原型(Prototype)模式的定义如下:用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型相同或相似的新对象。在这里,*      原型实例指定了要创建的对象的种类。用这种方式创建对象非常高效,根本无须知道对象创建的细节。例如,Windows 操作系统的安装通常较耗时,*      如果复制就快了很多。在生活中复制的例子非常多,这里不一一列举了。*      原型模式的结构与实现由于 Java 提供了对象的 clone() 方法,所以用 Java 实现原型模式很简单。*  模式的结构*      原型模式包含以下主要角色。*      抽象原型类:规定了具体原型对象必须实现的接口。   (Cloneable)*      具体原型类:实现抽象原型类的 clone() 方法,它是可被复制的对象。  (Prototype)*      访问类:使用具体原型类中的 clone() 方法来复制新的对象。  (Main.class)*/
public class Main {@Testpublic void prototypeTest() throws CloneNotSupportedException {Member initMember = new Member("c#");ArrayList<String> initHobbies = new ArrayList<>();initHobbies.add("play");Prototype prototype = new Prototype("张三",18,initHobbies,initMember);System.out.println("复制前");System.out.println("被克隆对象:"+prototype);Prototype prototype1 = (Prototype) prototype.clone();System.out.println("复制后");System.out.println("是否同一对象:"+(prototype==prototype1?"是":"否"));System.out.println("修改被克隆对象属性:");prototype.setName("李四");initHobbies.add("eat");initMember.setPosition("java");System.out.println("被克隆对象:"+prototype);System.out.println("克隆出的对象:"+prototype1);}}
  • 当把深克隆的实现方法注释,使用浅克隆的方法时

image.png

运行结果:


通过构造方法创建对象。。。。。。
复制前
被克隆对象:Prototype{name='张三', age=18, hobbies=[play], member=Member{position='c#'}}
复制后
是否同一对象:否
修改被克隆对象属性:
被克隆对象:Prototype{name='李四', age=18, hobbies=[play, eat], member=Member{position='java'}}
克隆出的对象:Prototype{name='张三', age=18, hobbies=[play, eat], member=Member{position='java'}}

  当修改被克隆对象hobbies与member属性的时候,克隆出来对象的hobbies与member属性也被修改,可见这两个属性都是同一对象引用,而String类型的name以及基础类型的age属性是不会被同时修改的,可见不是同一引用。浅克隆只复制对象的String类型属性以及一些基本类型属性,是不完全克隆。

  • 当把浅克隆的实现方法注释,使用深克隆的方法时

image.png

运行结果:


通过构造方法创建对象。。。。。。
复制前
被克隆对象:Prototype{name='张三', age=18, hobbies=[play], member=Member{position='c#'}}
复制后
是否同一对象:否
修改被克隆对象属性:
被克隆对象:Prototype{name='李四', age=18, hobbies=[play, eat], member=Member{position='java'}}
克隆出的对象:Prototype{name='张三', age=18, hobbies=[play], member=Member{position='c#'}}

  当修改被克隆对象name、age、hobbies以及member属性的时候,克隆出来对象的hobbies与member属性并未被修改,可见通过这种深克隆的方法,把所有的属性都创建一个新的内存对象,并使被克隆对象与克隆出的对象所有属性有不同的地址引用,深克隆的复制更加彻底。

  注意:上面代码在执行克隆的时候并未打印构造方法中的:“通过构造方法创建对象。。。。。。”,因此可见,使用原型模式复制对象不会调用类的构造方法。因为对象的复制是通过调用Object类的clone方法来完成的,它直接在内存中复制数据,因此不会调用到类的构造方法。不但构造方法中的代码不会执行,甚至连访问权限都对原型模式无效。还记得单例模式吗?单例模式中,只要将构造方法的访问权限设置为private型,就可以实现单例。但是clone方法直接无视构造方法的权限,所以,单例模式与原型模式是冲突的,在使用时要特别注意。

三、应用场景

  原型模式通常适用于以下场景。

  • 对象之间相同或相似,即只是个别的几个属性不同的时候。
  • 创建对象成本较大,例如初始化时间长,占用CPU太多,或者占用网络资源太多等,需要优化资源。
  • 创建一个对象需要繁琐的数据准备或访问权限等,需要提高性能或者提高安全性。
  • 系统中大量使用该类对象,且各个调用者都需要给它的属性重新赋值。

四、优缺点分析

  原型模式的优点

  • Java 自带的原型模式基于内存二进制流的复制,在性能上比直接 new 一个对象更加优良。
  • 可以使用深克隆方式保存对象的状态,使用原型模式将对象复制一份,并将其状态保存起来,简化了创建对象的过程,以便在需要的时候使用(例如恢复到历史某一状态),可辅助实现撤销操作。

  原型模式的缺点

  • 需要为每一个类都配置一个 clone 方法
  • clone 方法位于类的内部,当对已有类进行改造的时候,需要修改代码,违背了开闭原则。
  • 当实现深克隆时,需要编写较为复杂的代码,而且当对象之间存在多重嵌套引用时,为了实现深克隆,每一层对象对应的类都必须支持深克隆,实现起来会比较麻烦。因此,深克隆、浅克隆需要运用得当。

代码地址:https://gitee.com/fluffycatkin/JavaDesignModel.git

image.png

原文出处:
https://blog.csdn.net/zhengzhb/article/details/7393528
http://c.biancheng.net/view/1343.html

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

相关文章:

  • transformer实现词性标注
  • Java中异或操作和OTP算法
  • K8S最新版本集群部署(v1.28) + 容器引擎Docker部署(下)
  • 女子垒球运动的发展·垒球1号位
  • Debian 30 周年,生日快乐!
  • 字符串匹配的Rabin–Karp算法
  • 傅里叶变换(FFT)笔记存档
  • ELK安装、部署、调试 (二) ES的安装部署
  • Android 13 - Media框架(8)- MediaExtractor
  • Flutter 混合开发调试
  • C语言每日一练------(Day3)
  • 14、监测数据采集物联网应用开发步骤(10)
  • Linux禅道上修改Apache 和 MySQL 默认端口号
  • 操作教程|通过1Panel开源Linux面板快速安装DataEase
  • 机器学习策略——优化深度学习系统
  • ES6中Proxy和Proxy实例
  • UDP协议的重要知识点
  • QT6为工程添加资源文件,并在ui界面引用
  • Python小知识 - 如何使用Python的Flask框架快速开发Web应用
  • Flutter 项目结构文件
  • 未找到System.Runtime.InteropServices.Marshal.GetTypeFromCLSID(System.Guid) 方法错误
  • 人员位置管理,点亮矿山安全之路
  • node-red - 读写操作redis
  • 【图像处理】模板匹配的学习笔记
  • Ext JS之Ext Direct快速入门
  • 内网隧道技术学习
  • 【前端】CSS3新特性
  • Spring之HandlerInterceptor和RequestBodyAdvice
  • transition、transform 区别和应用
  • Android中级——消息机制