原型模式--深拷贝和浅拷贝

news/2024/11/24 0:15:44/

定义

Specify the kind of objects to create using a prototypical instance, and create new objects by copying this prototype. (使用原型实例指定将要创建的对象类型,通过复制这个实例创建新的对象。)

从定义中我们我们可以发现,该模式的前提是首先需要有一个对象,然后基于已有的对象复制生成新的对象,新生成的对象状态和原对象是一样的,不一样的是内存地址。

实现

public interface ICloneable<T> extends Cloneable {public T copy();
}

public class Message implements ICloneable<Message>{int what;String descriotion;@Overridepublic Message copy() {Message clone = null;try {clone = (Message) clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return clone;}
}

上述只有两个类,一个是ICloneable接口,该接口继承自java.lang.Cloneable接口,Cloneable接口是一个标记接口,本身没有什么方法,但是调用某个对象的clone方法必须要实现该接口,这里我借助该接口并扩展了ICloneable接口,所有ICloneable接口的实现类都必须要具备复制自己的能力,你可以发现,当前的设计中没有要求实现者必须是深拷贝或者浅拷贝。此外这里的接口作为一个标记来处理的方式很精巧,在日常开发中我们也可以这样做,比如设计一个防混淆的接口,所有该接口的实现者都在proguard中配置不混淆。
第二个类是ICloneable实现者,其copy方法对外提供了复制自身的能力,Message类中的what变量属于基本数据类型,该变量会被直接复制后赋值给新对象,而description变量是对象类型,目前copy方法采用了默认处理(clone方法默认是浅拷贝),所以该变量仍然指向旧对象中decscription变量所指向的string对象。

扩展

在解释原型模式的定义时,我们说原型对象是基于已有对象进行复制来生成新的对象,那么新生成对象时会执行构造函数吗?
在Java中,创建一个堆上对象有五种方式:

  • 使用new关键字 → 调用了构造函数
  • 使用Class类的newInstance方法 → 调用了构造函数
  • 使用Constructor类的newInstance方法 → 调用了构造函数
  • 使用clone方法 → 没有调用构造函数
  • 使用反序列化 → 没有调用构造函数

clone的行为是很简单的。以堆上的内存存储解释的话(不计内务内存),对一个对象a的clone就是在堆上分配一个和a在堆上所占存储空间一样大的一块地方,然后把a的堆上内存的内容复制到这个新分配的内存空间上。

浅拷贝(Shallow Clone):当原型对象被复制时,只复制它本身和其中包含的值类型的成员变量,而引用类型的成员变量并没没复制。如我想将A复制一份出来,命名为为B,那么我在浅拷贝后,确实可以得到A和B。而且A和B的值也相等,但是,我将B的值稍作修改,A的值也会变动,这往往不是我们想要的。因为我们想拷贝一个副本出来,二者也能独立,这样才算拷贝。但是,浅拷贝后A和B却是指向同一片地址空间,也就是二者共用一个值,改一个,两个都变。

浅拷贝实现:

重写Object类中的clone方法。Java中,所有类的父类都是Object类,Object类中有一个clone方法,作用是返回对象的一个拷贝,但是其作用域protected类型的,一般的类无法调用,因此,需要拷贝的类需要将clone方法的作用域修改为public类型。

注:Object类的clone方法只会拷贝java中的8中基本类型以及他们的封装类型,另外还有String类型。对于数组、容器对象、引用对象等都不会拷贝,这就是浅拷贝。如果要实现深拷贝,必须将原型模式中的数组、容器对象、引用对象等另行拷贝。

public class Student implements Cloneable {private String name;private int age;private Address addr;public Student(String name, int age, Address addr) {this.name = name;this.age = age;this.addr = addr;}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public Address getAddr() {return addr;}public void setAddr(Address addr) {this.addr = addr;}@Overridepublic String toString() {return "Student{" +"name='" + name + '\'' +", age=" + age +", addr=" + addr +'}';}@Overridepublic Object clone() throws CloneNotSupportedException {return super.clone();}
}class Address implements Cloneable {private String addrName;public Address(String addrName) {this.addrName = addrName;}public String getAddrName() {return addrName;}public void setAddrName(String addrName) {this.addrName = addrName;}@Overridepublic String toString() {return "Address{" +"addrName='" + addrName + '\'' +'}';}@Overridepublic Object clone() throws CloneNotSupportedException {return super.clone();}
}class Test {public static void main(String[] args) throws CloneNotSupportedException {Student student = new Student("zhangsan", 13, new Address("China"));Student clone = (Student) student.clone();System.out.println(student);System.out.println(clone);student.setName("lisi");student.getAddr().setAddrName("Canada");System.out.println(student);System.out.println(clone);}
}

深拷贝(Deep Clone):除了对象本身被复制外,对象所包含的所有成员变量也被复制,就是我们想要的那种拷贝,即有一个副本,与原者老死不想往来,互不影响。

把上面的Student中clone方法改造一下,即可以实现深拷贝:

@Override
public Object clone() throws CloneNotSupportedException {Student newStu = (Student) super.clone();newStu.addr = (Address) this.getAddr().clone();return newStu;
}

拷贝的时候,将对象中的引用数据也拷贝一份即可。

2、可以使用序列化 + 反序列化的方式实现深拷贝:

public Object deepCopy(Object object) {ByteArrayOutputStream bo = new ByteArrayOutputStream();ObjectOutputStream oo = new ObjectOutputStream(bo);oo.writeObject(object);ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());ObjectInputStream oi = new ObjectInputStream(bi);return oi.readObject();
}

 

原型模式的优点:

  1. 简化对象的创建过程,通过复制一个已有对象实例可以提高新实例的创建效率
  2. 扩展性好
  3. 提供了简化的创建结构,原型模式中的产品的复制是通过封装在原型类中的克隆方法实现的,无需专门的工厂类来创建产品
  4. 可以通过深克隆的方式保存对象的状态,使用原型模式将对象复制一份并其状态保存起来,以便在需要的时候使用,可辅助实现撤销操作
  • 原型模式的缺点:
  1. 需要为每一个类准备一个克隆方法,而且该克隆方法位于一个类的内部,当对已有类进行改造时,需要修改原代码,违背了开闭原则
  2. 在实现深克隆时需要写较复杂的代码,而且当对象之间存在多重的嵌套引用时,为了实现深克隆,每一层对象对应的类必须支持深克隆,实现起来较烦较烦....
  • 原型模式的适用环境:
  1. 创建新对象成本较大,新对象可以通过复制已有对象来获得,如果是相似对象,则可以对其成员变量修改
  2. 系统要保存对象的状态,而对象的状态变化很小
  3. 需要避免使用分层次的工厂类来创建分层次的对象,并且类的实例对象只有一个或很少的几个组合状态,通过复制原型对象得到新实例可能比使用构造函数创建一个新实例更方便

http://www.ppmy.cn/news/63649.html

相关文章

基数树RadixTree

转自&#xff1a;基数树RadixTree - 知乎 1. 基数树概述 对于长整型数据的映射&#xff0c;如何解决Hash冲突和Hash表大小的设计是一个很头疼的问题。radix树就是针对这种稀疏的长整型数据查找&#xff0c;能快速且节省空间地完成映射。借助于Radix树&#xff0c;我们可以实现…

SpringBoot整合接口管理工具Swagger

Swagger Swagger简介 Springboot整合swagger Swagger 常用注解 一、Swagger简介 ​ Swagger 是一系列 RESTful API 的工具&#xff0c;通过 Swagger 可以获得项目的⼀种交互式文档&#xff0c;客户端 SDK 的自动生成等功能。 ​ Swagger 的目标是为 REST APIs 定义一个标…

嵌入式软考备考_5 安全性基础知识

安全性基础知识 网安问题概述 被动攻击&#xff1a;监听&#xff08;截获&#xff09;。 主动攻击&#xff1a;主动破坏&#xff08;中断篡改&#xff0c;病毒&#xff0c;ddos使得某个服务拒绝服务&#xff0c;重放攻击&#xff1a;黑客截取了正常用户输入用户名密码的加密…

Spring Boot 整合 Swagger 教程详解

✅作者简介&#xff1a;2022年博客新星 第八。热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏…

3. SQL底层执行原理详解

一条SQL在MySQL中是如何执行的 1. MySQL的内部组件结构1.1 Server层1.2 Store层 2. 连接器3. 分析器4. 优化器5. 执行器6. bin-log归档 本文是按照自己的理解进行笔记总结&#xff0c;如有不正确的地方&#xff0c;还望大佬多多指点纠正&#xff0c;勿喷。 1. MySQL的内部组件结…

python中使用opencv LED屏数字识别(可用做车牌识别,一样的原理)

应项目要求需要基于cpu的LED数字识别&#xff0c;为了满足需求&#xff0c;使用传统方法进行实验。识别传感器中显示的数字。因此使用opencv的函数做一些处理&#xff0c;实现功能需求。 首先读取图像&#xff0c;因为我没想大致得到LED屏幕的区域&#xff0c;因此将RGB转换为H…

【JavaScript由浅入深】常用的正则表达式

【JavaScript由浅入深】常用的正则表达式 文章目录 【JavaScript由浅入深】常用的正则表达式写在前面一、认识正则表达式1.1 正则表达式的概念 二、正则表达式的使用2.1 使用构造函数创建正则表达式2.2 使用字面量创建正则表达式2.3 补充 三、正则表达式常见规则3.1 字符类3.2 …

Oracle imp/impdp、exp/expdp的使用方法

以下是创建新用户并授权角色和系统权限&#xff0c;使用imp/impdp导入dmp文件到数据库&#xff0c;exp/expdp导出文件到数据库的综合示例&#xff1a; 创建新用户并授权角色和系统权限 CREATE USER new_user IDENTIFIED BY password;GRANT CONNECT, RESOURCE TO new_user;GRA…