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

Java设计模式 —— 【创建型模式】原型模式(浅拷贝、深拷贝)详解

文章目录

  • 前言
  • 原型模式
  • 一、浅拷贝
      • 1、案例
      • 2、引用数据类型
  • 二、深拷贝
      • 1、重写clone()方法
      • 2、序列化
  • 总结


前言

先看一下传统的对象克隆方式:

原型类:

public class Student {private String name;public Student(String name) {this.name = name;}public String getName() {return name;}@Overridepublic String toString() {return "Student{'name' = " + name + "}, " + "hashCode = " + this.hashCode();}
}

克隆:

@Test
public void test(){//原型对象Student student = new Student("张三");//克隆对象Student student1 = new Student(student.getName());Student student2 = new Student(student.getName());Student student3 = new Student(student.getName());System.out.println("原型对象: " + student);System.out.println("克隆对象1: " + student1);System.out.println("克隆对象2: " + student2);System.out.println("克隆对象3: " + student3);
}

在这里插入图片描述

  1. 优点是比较好理解,简单易操作;
  2. 在创建新的对象时,总是需要重新获取原始对象的属性,如果创建的对象比较复杂时,效率较低;
  3. 总是需要重新初始化对象,而不是动态地获得对象运行时的状态, 不够灵活。

原型模式

  1. 原型模式(Prototype模式)是指:用原型实例指定创建对象的种类,并且通过拷贝这些原型,创建新的对象;
  2. 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象,无需知道如何创建的细节;
  3. 工作原理是: 通过将一个原型对象传给那个要发动创建的对象,这个要发动创建的对象通过请求原型对象拷贝它们自己来实施创建,即 对象.clone()

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

原型模式包含如下角色:

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

在这里插入图片描述
原型模式的克隆分为浅克隆和深克隆。

  • 浅克隆:创建一个新对象,新对象的属性和原来对象完全相同,对于非基本类型属性,仍指向原有属性所指向的对象的内存地址。
  • 深克隆:创建一个新对象,属性中引用的其他对象也会被克隆,不再指向原有对象地址。

一、浅拷贝

1、案例

对于上文中的克隆方法加以改进:

原型类:

public class Student implements Cloneable {private String name;public Student(String name) {System.out.println("原型对象创建成功!!!");this.name = name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "Student{'name' = " + name + "}, " + "hashCode = " + this.hashCode();}//实现对象克隆@Overrideprotected Object clone() throws CloneNotSupportedException {System.out.println("克隆成功!!!");return super.clone();}
}

测试:

@Test
public void test1() throws CloneNotSupportedException {Student newStudent = new Student("张三");Student cloneStudent = (Student) newStudent.clone();System.out.println("原型对象: " + newStudent);System.out.println("克隆对象: " + cloneStudent);
}

在这里插入图片描述

2、引用数据类型

  • 上述案例中我们可以看出克隆是克隆成功了,并且没有走构造方法,所克隆出的对象地址和原对象地址不一样,是新的对象;

  • 对于数据类型是基本数据类型的成员变量,浅拷贝会直接进行值传递,也就是将该属性值复制一份给新的对象;

  • 但是引用数据类型的成员变量,比如说成员变量是某个数组、某个类的对象等,并没有new 一个新的对象,而是进行引用传递指向原有的引用;

  • 在这种情况下,在一个对象中修改该成员变量会影响到另一个对象的该成员变量值。

我们添加原型类的成员变量:

School:

public class School {private String name;public School(String name) {this.name = name;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}

Student:

public class Student implements Cloneable {private String name;private School school;public Student(String name, School school) {this.name = name;this.school = school;}public void setName(String name) {this.name = name;}public School getSchool() {return school;}@Overridepublic String toString() {return "Student{'name' = " + name + ", 'school' = " + school.getName() + "}, " +"Student.hashCode = " + this.hashCode() + ", " +"name.hashCode" + name.hashCode() + ", " +"School.hashCode = " + school.hashCode();}//实现对象克隆@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

测试:

@Test
public void test2() throws CloneNotSupportedException {Student newStudent = new Student("张三", new School("清华"));Student cloneStudent = (Student) newStudent.clone();System.out.println("原型对象:" + newStudent);System.out.println("克隆对象:" + cloneStudent);System.out.println("=====================修改克隆对象信息========================");cloneStudent.setName("李四");cloneStudent.getSchool().setName("北大");System.out.println("修改后的原型对象:" + newStudent);System.out.println("修改后的克隆对象:" + cloneStudent);
}

在这里插入图片描述
上述案例可以看出:

  • 克隆确实产生新的对象,但是引用数据类型只是进行了引用传递;
  • 以至于我们修改了cloneStudent的学校,newStudent也随之修改了;
  • 那为什么String也是引用数据类型,cloneStudent的那么由“张三”改为“李四”,而newStudent没有呢,那是因为String不可变,传入新的,当然指向新的地址了。

二、深拷贝

  1. 复制对象的所有基本数据类型的成员变量值

  2. 为所有引用数据类型的成员变量申请存储空间,并复制每个引用数据类型成员量所引用的对象,直到该对象可达的所有对象。也就是说,对象进行深拷贝要对整个对象进行拷贝

  3. 深拷贝实现方式有两种
    - 重写clone方法来实现深拷贝

    - 通过对象序列化实现深拷贝(推荐)

1、重写clone()方法

  • 重写clone方法主要是在原有的克隆的基础上,将引用数据类型再进行嵌套克隆;

  • 每个被引用的类也要实现Cloneable接口,重写clone()方法;

  • 这对全新的类来说不是很难,但对已有的类进行改造时,需要修改其源代码,违背开闭原则。

School:

public class School implements Cloneable{private String name;public School(String name) {this.name = name;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}
}

Student:

public class Student implements Cloneable {private String name;private School school;public Student(String name, School school) {this.name = name;this.school = school;}public void setName(String name) {this.name = name;}public School getSchool() {return school;}@Overridepublic String toString() {return "Student{'name' = " + name + ", 'school' = " + school.getName() + "}, " +"Student.hashCode = " + this.hashCode() + ", " +"name.hashCode" + name.hashCode() + ", " +"School.hashCode = " + school.hashCode();}//实现对象克隆@Overrideprotected Object clone() throws CloneNotSupportedException {//克隆基本数据类型以及StringStudent student = (Student) super.clone();//引用数据类型再进行克隆student.school = (School) student.getSchool().clone();return student;}
}

测试:

@Test
public void test3() throws CloneNotSupportedException {Student newStudent = new Student("张三", new School("清华"));Student cloneStudent = (Student) newStudent.clone();System.out.println("原型对象:" + newStudent);System.out.println("克隆对象:" + cloneStudent);System.out.println("=====================修改克隆对象信息========================");cloneStudent.setName("李四");cloneStudent.getSchool().setName("北大");System.out.println("修改后的原型对象:" + newStudent);System.out.println("修改后的克隆对象:" + cloneStudent);
}

在这里插入图片描述

2、序列化

涉及到的所有类必须实现Serializable接口,否则会抛NotSerializableException异常。

School:

public class School implements Serializable{private String name;public School(String name) {this.name = name;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}

Student:

public class Student implements Serializable {private String name;private School school;public Student(String name, School school) {this.name = name;this.school = school;}public void setName(String name) {this.name = name;}public School getSchool() {return school;}@Overridepublic String toString() {return "Student{'name' = " + name + ", 'school' = " + school.getName() + "}, " +"Student.hashCode = " + this.hashCode() + ", " +"name.hashCode" + name.hashCode() + ", " +"School.hashCode = " + school.hashCode();}public Student 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);return (Student) ois.readObject();} catch (Exception e) {e.printStackTrace();return null;} finally {try {if (bos != null) bos.close();if (oos != null) oos.close();if (bis != null) bis.close();if (ois != null) ois.close();} catch (IOException e) {e.printStackTrace();}}}
}

测试:

@Test
public void test4() throws CloneNotSupportedException {Student newStudent = new Student("张三", new School("清华"));Student cloneStudent = newStudent.deepClone();System.out.println("原型对象:" + newStudent);System.out.println("克隆对象:" + cloneStudent);System.out.println("=====================修改克隆对象信息========================");cloneStudent.setName("李四");cloneStudent.getSchool().setName("北大");System.out.println("修改后的原型对象:" + newStudent);System.out.println("修改后的克隆对象:" + cloneStudent);
}

在这里插入图片描述


总结

原型模式的注意事项和细节:

  1. 创建新的对象比较复杂时,可以利用原型模式简化对象的创建过程,同时也能够提高效率;
  2. 不用重新初始化对象,而是动态地获得对象运行时的状态;
  3. 如果原始对象发生变化(增加或者减少属性),其它克隆对象的也会发生相应的变化,无需修改代码;
  4. 需要注意浅拷贝的成员变量数据类型是引用数据类型(对象)的时候;
  5. 在实现深克隆的时候可能需要比较复杂的代码建议使用序列化方式;

相关文章:

Java设计模式 —— 【创建型模式】原型模式(浅拷贝、深拷贝)详解

文章目录 前言原型模式一、浅拷贝1、案例2、引用数据类型 二、深拷贝1、重写clone()方法2、序列化 总结 前言 先看一下传统的对象克隆方式: 原型类: public class Student {private String name;public Student(String name) {this.name name;}publi…...

SciAssess——评估大语言模型在科学文献处理中关于模型的记忆、理解和分析能力的基准

概述 大规模语言模型(如 Llama、Gemini 和 GPT-4)的最新进展因其卓越的自然语言理解和生成能力而备受关注。对这些模型进行评估对于确定其局限性和潜力以及促进进一步的技术进步非常重要。为此,人们提出了一些特定的基准来评估大规模语言模型…...

SQLModel与FastAPI结合:构建用户增删改查接口

SQLModel简介 SQLModel是一个现代化的Python库,旨在简化与数据库的交互。它结合了Pydantic和SQLAlchemy的优势,使得定义数据模型、进行数据验证和与数据库交互变得更加直观和高效。SQLModel由FastAPI的创始人Sebastin Ramrez开发,专为与FastA…...

【RISC-V CPU debug 专栏 2.3 -- Run Control】

文章目录 Run ControlHart 运行控制状态位状态信号操作流程时间与实现注意事项Run Control 在 RISC-V 调试架构中,运行控制模块通过管理多个状态位来对硬件线程(harts)的执行进行调节和控制。这些状态位帮助调试器请求暂停或恢复 harts,并在 hart 复位时进行控制。以下是运…...

探索 IntelliJ IDEA 中 Spring Boot 运行配置

前言 IntelliJ IDEA 作为一款功能强大的集成开发环境(IDE),为 Spring Boot 应用提供了丰富的运行配置选项,定义了如何在 IntelliJ IDEA 中运行 Spring Boot 应用程序,当从主类文件运行应用程序时,IDE 将创建…...

三除数枚举

给你一个整数 n 。如果 n 恰好有三个正除数 ,返回 true ;否则,返回 false 。 如果存在整数 k ,满足 n k * m ,那么整数 m 就是 n 的一个 除数 。 输入:n 4 输出:true 解释:4 有三…...

【051】基于51单片机温度计【Proteus仿真+Keil程序+报告+原理图】

☆、设计硬件组成:51单片机最小系统DS18B20温度传感器LCD1602液晶显示按键设置蜂鸣器LED灯。 1、本设计采用STC89C51/52、AT89C51/52、AT89S51/52作为主控芯片; 2、采用DS18B20温度传感器测量温度,并且通过LCD1602实时显示温度;…...

[Java]微服务之服务保护

雪崩问题 微服务调用链路中的某个服务故障,引起整个链路中的所有微服务都不可用,这就是雪崩 雪崩问题产生的原因是什么? 微服务相互调用,服务提供者出现故障或阻塞。服务调用者没有做好异常处理,导致自身故障。调用链中的所有服…...

自动驾驶目标检测融合全貌

1、early fusion 早期融合,特点用到几何空间转换3d到2d或者2d到3d的转换,用像素找点云或者用点云找像素。 2、deep fusion 深度融合,也是特征级别融合,也叫多模态融合,如bevfusion范式 3、late fusion 晚融合&#x…...

消息框(Message Box)的测试方法和测试用例

我来帮你了解消息框(Message Box)的测试方法和测试用例的编写。 我已经创建了一个测试用例示例,让我为你解释消息框测试的主要方面: 测试维度: 功能性测试:验证消息框的基本功能是否正常样式测试:确认不同类型消息框…...

Ubuntu 包管理

APT&dpkg 查看已安装包 查看所有已经安装的包 dpkg -l 查找包 apt search <package_name>搜索软件包列表&#xff0c;找到与搜索关键字匹配的包 dpkg与grep结合查找特定的包 dpkg -s <package>&#xff1a;查看某个安装包的详细信息 安装包 apt安装命令 更新…...

[Ubuntu] linux之Ubuntu18.04的下载及在虚拟机中详细安装过程(附有下载链接)

前言 ubuntu 链接&#xff1a;https://pan.quark.cn/s/283509d0d36e 提取码&#xff1a;dfT1 链接失效&#xff08;可能被官方和谐&#xff09;可评论或私信我重发 下载压缩包后解压 &#xff01;&#xff01;安装路径不要有中文 下载后解压得到.iso文件&#xff0c;不要放在…...

ffmpeg安装(windows)

ffmpeg安装-windows 前言ffmpeg安装路径安装说明 前言 ffmpeg的安装也是开箱即用的,并没有小码哥说的那么难 ffmpeg安装路径 这就下载好了! 安装说明 将上面的bin目录加入到环境变量,然后在cmd中测试一下: C:\Users\12114\Desktop\test\TaskmgrPlayer\x64\Debug>ffmpe…...

服务器数据恢复—raid6阵列硬盘被误重组为raid5阵列的数据恢复案例

服务器存储数据恢复环境&#xff1a; 存储中有一组由12块硬盘组建的RAID6阵列&#xff0c;上层linux操作系统EXT3文件系统&#xff0c;该存储划分3个LUN。 服务器存储故障&分析&#xff1a; 存储中RAID6阵列不可用。为了抢救数据&#xff0c;运维人员使用原始RAID中的部分…...

linux内核编译启动总结

linux kernel 编译 升级汇总 写在前面内核编译获取kernel代码开始前的准备工作 编译过程1\.解压与净化将下载好的linux内核解压至/usr/src 2\. 得到源代码后,将其净化3\. 配置要进行编译的内核4.编译内核. &#xff08;15分钟&#xff09;5.编译模块.方法1:方法2&#xff1a; 6…...

Android Studio的AI工具插件使用介绍

Android Studio的AI工具插件使用介绍 一、前言 Android Studio 的 AI 工具插件具有诸多重要作用&#xff0c;以下是一些常见的方面&#xff1a; 代码生成与自动补全 代码优化与重构 代码解读 学习与知识获取 智能搜索与资源推荐实际使用中可以添加注释&#xff0c;解读某段代…...

本地部署 WireGuard 无需公网 IP 实现异地组网

WireGuard 是一个高性能、极简且易于配置的开源虚拟组网协议。使用路由侠内网穿透使其相互通讯。 第一步&#xff0c;服务端&#xff08;假设为公司电脑&#xff09;和客户端&#xff08;假设为公司外的电脑&#xff09;安装部署 WireGuard 1&#xff0c;点此下载&#xff08;…...

asyncio.ensure_future 与 asyncio.create_task:Python异步编程中的选择

asyncio.ensure_future 与 asyncio.create_task&#xff1a;Python异步编程中的选择 引言asyncio.ensure_futureasyncio.create_task两者的区别参数接受范围任务调度的保证代码可读性 哪个更好&#xff1f;使用asyncio.create_task使用asyncio.ensure_future 结论参考 引言 在…...

CTF之密码学(密码特征分析)

一.MD5,sha1,HMAC,NTLM 1.MD5&#xff1a;MD5一般由32/16位的数字(0-9)和字母(a-f)组成的字符串 2.sha1&#xff1a;这种加密的密文特征跟MD5差不多&#xff0c;只不过位数是40&#xff08;sha256&#xff1a;64位&#xff1b;sha512:128位&#xff09; 3.HMAC&#xff1a;这…...

JVM调优篇之JVM基础入门AND字节码文件解读

目录 Java程序编译class文件内容常量池附录-访问标识表附录-常量池类型列表 Java程序编译 Java文件通过编译成class文件后&#xff0c;通过JVM虚拟机解释字节码文件转为操作系统执行的二进制码运行。 规范 Java虚拟机有自己的一套规范&#xff0c;遵循这套规范&#xff0c;任…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件

今天呢&#xff0c;博主的学习进度也是步入了Java Mybatis 框架&#xff0c;目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学&#xff0c;希望能对大家有所帮助&#xff0c;也特别欢迎大家指点不足之处&#xff0c;小生很乐意接受正确的建议&…...

电脑插入多块移动硬盘后经常出现卡顿和蓝屏

当电脑在插入多块移动硬盘后频繁出现卡顿和蓝屏问题时&#xff0c;可能涉及硬件资源冲突、驱动兼容性、供电不足或系统设置等多方面原因。以下是逐步排查和解决方案&#xff1a; 1. 检查电源供电问题 问题原因&#xff1a;多块移动硬盘同时运行可能导致USB接口供电不足&#x…...

2021-03-15 iview一些问题

1.iview 在使用tree组件时&#xff0c;发现没有set类的方法&#xff0c;只有get&#xff0c;那么要改变tree值&#xff0c;只能遍历treeData&#xff0c;递归修改treeData的checked&#xff0c;发现无法更改&#xff0c;原因在于check模式下&#xff0c;子元素的勾选状态跟父节…...

【JavaWeb】Docker项目部署

引言 之前学习了Linux操作系统的常见命令&#xff0c;在Linux上安装软件&#xff0c;以及如何在Linux上部署一个单体项目&#xff0c;大多数同学都会有相同的感受&#xff0c;那就是麻烦。 核心体现在三点&#xff1a; 命令太多了&#xff0c;记不住 软件安装包名字复杂&…...

ArcGIS Pro制作水平横向图例+多级标注

今天介绍下载ArcGIS Pro中如何设置水平横向图例。 之前我们介绍了ArcGIS的横向图例制作&#xff1a;ArcGIS横向、多列图例、顺序重排、符号居中、批量更改图例符号等等&#xff08;ArcGIS出图图例8大技巧&#xff09;&#xff0c;那这次我们看看ArcGIS Pro如何更加快捷的操作。…...

Redis的发布订阅模式与专业的 MQ(如 Kafka, RabbitMQ)相比,优缺点是什么?适用于哪些场景?

Redis 的发布订阅&#xff08;Pub/Sub&#xff09;模式与专业的 MQ&#xff08;Message Queue&#xff09;如 Kafka、RabbitMQ 进行比较&#xff0c;核心的权衡点在于&#xff1a;简单与速度 vs. 可靠与功能。 下面我们详细展开对比。 Redis Pub/Sub 的核心特点 它是一个发后…...

代码随想录刷题day30

1、零钱兑换II 给你一个整数数组 coins 表示不同面额的硬币&#xff0c;另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额&#xff0c;返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带…...

无人机侦测与反制技术的进展与应用

国家电网无人机侦测与反制技术的进展与应用 引言 随着无人机&#xff08;无人驾驶飞行器&#xff0c;UAV&#xff09;技术的快速发展&#xff0c;其在商业、娱乐和军事领域的广泛应用带来了新的安全挑战。特别是对于关键基础设施如电力系统&#xff0c;无人机的“黑飞”&…...

Webpack性能优化:构建速度与体积优化策略

一、构建速度优化 1、​​升级Webpack和Node.js​​ ​​优化效果​​&#xff1a;Webpack 4比Webpack 3构建时间降低60%-98%。​​原因​​&#xff1a; V8引擎优化&#xff08;for of替代forEach、Map/Set替代Object&#xff09;。默认使用更快的md4哈希算法。AST直接从Loa…...

Razor编程中@Html的方法使用大全

文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...