原型模式

原型模式(Prototype Pattern),是指原型实例指定创建对象的种类,并且通过复制这些原型创建新的对象。

原型模式主要适用于以下场景:

  1. 类初始化消耗资源较多
  2. 使用new生成一个对象需要非常繁琐的过程(数据准备、访问权限等)
  3. 构造函数比较复杂
  4. 在循环中产生大量对象

浅拷贝

先创建原型Prototype接口:


/**
* @author ss_419
*/
public interface Prototype {
Prototype clone();
}

创建具体需要克隆的类ConcretePrototypeA:

package org.example.spring.designpattern.prototype.shallow;

import java.util.List;

/**
* TODO
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:15
*/
public class ConcretePrototypeA implements Prototype {
private int age;
private String name;
private List hobbies; public int getAge() {
return age;
} public void setAge(int age) {
this.age = age;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public List getHobbies() {
return hobbies;
} public void setHobbies(List hobbies) {
this.hobbies = hobbies;
} @Override
public Prototype clone() {
ConcretePrototypeA concretePrototypeA = new ConcretePrototypeA();
concretePrototypeA.setAge(this.getAge());
concretePrototypeA.setName(this.getName());
concretePrototypeA.setHobbies(this.getHobbies());
return concretePrototypeA;
} @Override
public String toString() {
return "ConcretePrototypeA{" +
"age=" + age +
", name='" + name + '\'' +
", hobbies=" + hobbies +
'}';
}
}

创建Client类:

package org.example.spring.designpattern.prototype.shallow;

/**
* TODO
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:17
*/
public class Client {
private Prototype prototype; public Client(Prototype prototype) {
this.prototype = prototype;
}
public Prototype startClone(Prototype concretePrototype){
return (Prototype) concretePrototype.clone();
}
}

测试代码如下:

package org.example.spring.designpattern.prototype.shallow;

import java.util.ArrayList;
import java.util.List; /**
* TODO
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:20
*/
public class PrototypeTest {
public static void main(String[] args) {
ConcretePrototypeA concretePrototypeA = new ConcretePrototypeA();
concretePrototypeA.setAge(18);
concretePrototypeA.setName("ss");
List hobbies = new ArrayList<String>();
hobbies.add("唱");
hobbies.add("跳");
hobbies.add("Rap");
concretePrototypeA.setHobbies(hobbies);
System.out.println(concretePrototypeA); // 创建client对象,准备开始克隆
Client client = new Client(concretePrototypeA);
ConcretePrototypeA concretePrototypeClone = (ConcretePrototypeA) client.startClone(concretePrototypeA);
System.out.println(concretePrototypeClone); System.out.println("concretePrototypeA.getHobbies() = " + concretePrototypeA.getHobbies());
System.out.println("concretePrototypeClone.getHobbies() = " + concretePrototypeClone.getHobbies());
System.out.println("对象地址比较:" +(concretePrototypeClone.getHobbies() == concretePrototypeA.getHobbies()));
}
}

浅拷贝只是完整复制了值类型数据,没有赋值引用对象,换言之,所有的引用对象仍然指向原来的对象,显然这不是我们想要的结果。

深拷贝

创建原型猴子类Monkey:

package org.example.spring.designpattern.prototype.deep;

import java.util.Date;

/**
* TODO 深拷贝对象——猴子
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:41
*/
public class Monkey {
public int height;
public int weight;
public Date birthDate;
}

创建引用对象JinGuBang:

package org.example.spring.designpattern.prototype.deep;

import java.io.Serializable;

/**
* TODO 引用对象JinGuBang
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:43
*/
public class JinGuBang implements Serializable {
public float h = 100;
public float d = 10; public void big(){
this.d *= 2;
this.h *= 2;
}
public void small(){
this.d /= 2;
this.h /= 2;
}
}

创建具体的对象齐天大圣类QiTianDaSheng:

package org.example.spring.designpattern.prototype.deep;

import java.io.*;
import java.util.Date; /**
* TODO 具体对象齐天大圣,归根结底还是个猴儿~
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:45
*/
public class QiTianDaSheng extends Monkey implements Cloneable, Serializable {
public JinGuBang jinGubang; public QiTianDaSheng() {
// 只是初始化
this.birthDate = new Date();
this.jinGubang = new JinGuBang();
} @Override
protected Object clone() throws CloneNotSupportedException {
// 调用指定的拷贝方法(深deep、浅shallow)
return this.deepClone();
} // 深拷贝
public Object deepClone() {
try{
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
// 写入当前对象
oos.writeObject(this); ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis); QiTianDaSheng copy = (QiTianDaSheng)ois.readObject();
copy.birthDate = new Date();
return copy;
}catch (Exception e){
e.printStackTrace();
return null;
}
} public QiTianDaSheng shallowClone(QiTianDaSheng target){
QiTianDaSheng qiTianDaSheng = new QiTianDaSheng();
qiTianDaSheng.height = target.height;
qiTianDaSheng.weight = target.weight; qiTianDaSheng.jinGubang = target.jinGubang;
qiTianDaSheng.birthDate = new Date(); return qiTianDaSheng;
}
}

测试代码如下:

package org.example.spring.designpattern.prototype.deep;

/**
* TODO 深拷贝测试代码
*
* @author ss_419
* @version 1.0
* @date 2023/3/9 09:53
*/
public class DeepCloneTest {
public static void main(String[] args) {
QiTianDaSheng qiTianDaSheng = new QiTianDaSheng();
try {
QiTianDaSheng clone
= (QiTianDaSheng) qiTianDaSheng.clone();
System.out.println("深拷贝:" + (qiTianDaSheng.jinGubang == clone.jinGubang));
}catch (Exception e){
e.printStackTrace();
}
QiTianDaSheng q = new QiTianDaSheng();
QiTianDaSheng n = q.shallowClone(q);
System.out.println("浅拷贝:" + (q.jinGubang == n.jinGubang)); }
}

运行结果如下:



我们可以发现,深拷贝的引用类对象是不同的指向,而浅拷贝的引用地址是相同的,这意味着深拷贝复制的是值,而不是引用地址了。

Spring设计模式——原型模式的更多相关文章

  1. 10. 星际争霸之php设计模式--原型模式

    题记==============================================================================本php设计模式专辑来源于博客(jymo ...

  2. PHP 设计模式 原型模式(Prototype)之深/浅拷贝

      看PHP 设计模式 原型模式(Prototype)时,衍生出一个扩展问题之 原型拷贝的浅拷贝和深拷贝问题(不管写Java还是写PHP还是写JS时都多多少少遇到过对象拷贝问题)   比如写前端页面时 ...

  3. PHP设计模式 原型模式(Prototype)

    定义 和工厂模式类似,用来创建对象.但实现机制不同,原型模式是先创建一个对象,采用clone的方式进行新对象的创建. 场景 大对象的创建. 优点 1.可以在运行时刻增加和删除产品 2.可以改变值或结构 ...

  4. 【设计模式】Java设计模式 - 原型模式

    [设计模式]Java设计模式 - 原型模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 原创作品,更多关注我CSDN: 一个有梦有戏的人 准备将博客园.CSDN一起 ...

  5. Spring中常见的设计模式——原型模式

    1.原型模式应用场景 当遇到大量耗费劳动力的 get,set赋值场景时,如下: public class SetGetParam { public void setParam(UserDto user ...

  6. C#设计模式-原型模式

    在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创建这样的类实例,这未免会增加创建类的复杂度和耗费更多的内存空间,因为这样在内存中分配 ...

  7. [工作中的设计模式]原型模式prototype

    一.模式解析 提起prototype,最近看多了js相关的内容,第一印象首先是js的原型 var Person=function(name){ this.name=name; } Person.pro ...

  8. 我的Java设计模式-原型模式

    "不好意思,我是卧底!哇哈哈哈~"额......自从写了上一篇的观察者模式,就一直沉浸在这个角色当中,无法自拨.昨晚在看<使徒行者2>,有一集说到啊炮仗哥印钞票,我去, ...

  9. C++设计模式——原型模式

    什么是原型模式? 在GOF的<设计模式:可复用面向对象软件的基础>中是这样说的:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象.这这个定义中,最重要的一个词是“拷贝”,也就 ...

  10. Java设计模式—原型模式

    原型设计模式是一种比较简单的设计模式,在项目中使用的场景非常多. 个人理解: 原型模式实现了对Java中某个对象的克隆功能,即该对象的类必须implements实现Cloneable接口来标识为可被克 ...

随机推荐

  1. C# 内存回收

    开发完成之后发现自己写的程序内存占用太高,找到如下解决方案 使用了一个timer每2s调用一次ClearMemory() #region 内存回收 [DllImport("kernel32. ...

  2. HCIP-进阶实验02-ISIS协议部署

    进阶实验02-ISIS协议部署 1 实验需求 设备 接口 IP地址 备注 R1 G0/0/0 10.1.123.1/24 R2 G0/0/0 10.1.123.2/24 G0/0/1 10.1.24. ...

  3. 记一次p标签内容不换行记录

    p标签内容l里面如果全部是英文,那么默认是不会换行的. 需要添加word-wrap: break-word; 属性 这样就会自动换行了

  4. continue的使用

    continue是终止本次循环,进行下一个循环如果要跳到最开始的循环abc : for(){ for(){ for(){ continue abc; } }}

  5. 简体Windows系统登录玩新枫之谷方法

    2022-2-23  游戏更新后Locale.Emulato失效,64位游戏使用新的工具Locale_Remulator,请自行到GitHub上搜索下载,使用方法与Locale.Emulato类似. ...

  6. 关于ERROR 1366 (HY000): Incorrect string value: '\xF0\x9F\x87\xA8\xF0\x9F...'报错的解决心得

    昨天发现MySQL数据库执行发生了报错 ERROR 1366 (HY000): Incorrect string value: '\xF0\x9F\x87\xA8\xF0\x9F...' 马上意识到应 ...

  7. 微信小程序使用echart图表不随着页面滚动

    1,问题描述 使用echarts时界面滑动时,图标不跟随滑动,浮在元素上方. 2,最简单的方法 在ec-canvas中添加,force-use-old-canvas="true", ...

  8. 使用python-docx提取word中的表格

    提取表格 import docx from docx import Document #导入库 path = '123.docx' #文件路径 document = Document(path) #读 ...

  9. 20220718 第七组 陈美娜 java

    如果把变量直接声明在类里:成员变量(全局变量)成员变量->属性 如果把变量声明在某个方法里:局部变量 public:访问权限修饰符,后面讲 void:没有返回值 run():方法名,标识符 {} ...

  10. pntia7-7 软硬车厢交替排列 (40 分)

    7-7 软硬车厢交替排列 (40 分) 设车辆厂生产了硬座车厢和软座车厢共n节(混合在一起),要求使用队列的基本操作,编程实现所有硬座车厢和所有软座车厢交替排列. 例如硬座车厢用H来表示,软座车厢用S ...