孙悟空的身外身法术使用了Java设计模式:原型模式
定义
原型模式属于对象的创建型模式,通过给出一个原型对象来指明所要创建的对象的类型,然后用复制这个原型对象的办法创建出更多同类型的对象,这就是原型模式的用意
java语言的构建模型直接支持原型模式,所有的JavaBean都继承自java.lang.Object,而且Object类提供了一个clone()方法,可以将一个JavaBean对象复制一份
但是这个JavaBean必须实现一个标识接口Cloneable,表明这个JavaBean支持复制。
如果一个对象没有实现这个接口却使用了clone()方法,Java编译器会抛出异常:CloneNotSupportedException
意图
用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象
主要解决问题
在运行期建立和删除原型
何时使用
如果一个系统的产品类是动态加载的,而且产品类具有一定的等级结构,这时可以采用原型模式
优缺点
优点:
- 允许动态的增加或者减少产品类
- 提供简化的创建结构
- 具有给一个应用软件动态加载新功能的能力
缺点:
- 每一个类都必须匹配一个克隆方法
- 必须实现 Cloneable 接口
结构
原型模式有两种表现形式:简单形式和登记形式
简单形式的原型模式
涉及的角色:
- 客户(Client)角色:客户类提出创建对象的请求
- 抽象原型(Prototype)角色:通常由一个Java接口或者Java抽象类实现,此角色给出所有的具体原型类所需要的接口
- 具体原型(ConcretePrototype)角色:被复制的对象,此角色需要实现抽象原型角色所要求的接口
源码如下:
public class Client {
private Prototype prototype;
public void operation(Prototype example) {
Prototype p = (Prototype) example.clone();
}
}
抽象原型角色声明了一个clone方法:
public interface Prototype extends Cloneable {
Object clone();
}
public class ConcretePrototype implements Prototype {
/** 克隆方法 */
@Override
public Object clone() {
try {
return super.clone();
} catch (CloneNotSupportedException e) {
return null;
}
}
}
登记形式的原型模式
涉及的角色:
- 客户端(Client)角色:客户端向管理员提出创建对象的请求
- 抽象原型(Prototype)角色:通常由一个Java接口或者Java抽象类实现,此角色给出所有的具体原型类所需要的接口
- 具体原型(ConcretePrototype)角色:被复制的对象,此角色需要实现抽象原型角色所要求的接口
- 原型管理器(PrototypeManager)角色:创建具体原型类的对象,并记录每一个被创建的对象
源码如下:
public interface Prototype extends Cloneable {
Object clone();
}
public class ConcretePrototype implements Prototype {
/** 克隆方法 */
@Override
public synchronized Object clone() {
Prototype temp = null;
try {
temp = (Prototype) super.clone();
return temp;
} catch (CloneNotSupportedException e) {
System.out.println("克隆失败!");
} finally {
return temp;
}
}
}
public class PrototypeManager {
private Vector vector = new Vector();
/** 增加一个新的对象 */
public void add(Prototype prototype) {
vector.add(prototype);
}
/** 取出聚集中的一个对象 */
public Prototype get(int index) {
return (Prototype) vector.get(index);
}
/** 给出聚集的大小 */
public int getSize() {
return vector.size();
}
}
public class Client {
private PrototypeManager manager;
private Prototype prototype;
public void registerPrototype() {
prototype = new ConcretePrototype();
Prototype cloneType = (Prototype) prototype.clone();
manager.add(cloneType);
}
}
两种形式比较
如果需要创建的原型对象数目比较少而且比较固定的话,可以采用简单形式,原型对象的引用由客户端保存
如果要创建的原型对象数目不固定的话,可以采用登记形式,在这种情况下,客户端不保存对原型对象的引用,这个任务由管理员对象来完成
在复制一个原型对象之前,客户端可以查看管理员对象中是否已经有一个满足要求的原型对象,如果有可以直接取出引用,如果没有客户端就需要自行复制此原型对象
浅克隆和深克隆
浅克隆:
被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用都仍然指向原来的对象,换而言之,浅克隆仅仅复制所考虑的对象,而不复制它所引用的对象
深克隆:
被复制对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的变量;那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。换而言之,深克隆把要复制的对象所引用的对象都复制一遍,而这种对被引用到的对象的复制叫做间接复制
孙悟空的身外身法术
孙悟空一根毫毛可以变幻出许多的大圣,由此场景熟悉原型模式
浅克隆实现
孙悟空本人:
public class TheGreatestSage {
private Monkey monkey = new Monkey();
public void change() {
//创建大圣本尊对象
Monkey copyMonkey;
for (int i=0; i<2000; i++) {
//克隆大圣本尊
copyMonkey = (Monkey) monkey.clone();
System.out.println("大圣本尊的生日:" + monkey.getBirthDate());
System.out.println("变化出来的大圣的生日:" + copyMonkey.getBirthDate());
System.out.println("大圣本尊== 变化出来的大圣:" + (monkey == copyMonkey));
System.out.println("大圣本尊的金箍棒== 变化出来的大圣的金箍棒:"
+ (monkey.getStaff() == copyMonkey.getStaff()));
}
}
public static void main(String[] args) {
TheGreatestSage sage = new TheGreatestSage();
sage.change();
}
}
具体原型角色:
public class Monkey implements Cloneable {
/** 身高 */
private float height;
/** 体重 */
private float weight;
/** 金箍棒 */
private GoldRingedStaff staff;
/** 日期 */
private Date birthDate;
public Monkey() {
this.birthDate = new Date();
}
/** 克隆方法 */
@Override
public Object clone() {
Monkey temp = null;
try {
temp = (Monkey) super.clone();
} catch (CloneNotSupportedException e) {
System.out.println("克隆失败!");
e.printStackTrace();
} finally {
return temp;
}
}
public float getHeight() {
return height;
}
public void setHeight(float height) {
this.height = height;
}
public float getWeight() {
return weight;
}
public void setWeight(float weight) {
this.weight = weight;
}
public Date getBirthDate() {
return birthDate;
}
public void setBirthDate(Date birthDate) {
this.birthDate = birthDate;
}
public GoldRingedStaff getStaff() {
return staff;
}
public void setStaff(GoldRingedStaff staff) {
this.staff = staff;
}
}
大圣拥有的金箍棒:
public class GoldRingedStaff {
/** 高度 */
private float height = 100.0f;
/** 半径 */
private float radius = 5.0f;
/** 构造函数 */
public GoldRingedStaff() {
}
/** 增长行为,每次调用高度和半径增加一倍 */
public void grow() {
this.height = this.height * 2;
this.radius = this.radius * 2;
}
/** 缩小行为,每次调用高度和半径减少一半 */
public void shrink() {
this.height = this.height / 2;
this.radius = this.radius / 2;
}
/** 移动 */
public void move() {
}
/** 高度取值方法 */
public float getHeight() {
return height;
}
/** 高度赋值方法 */
public void setHeight(float height) {
this.height = height;
}
/** 半径取值方法 */
public float getRadius() {
return radius;
}
/** 半径赋值方法 */
public void setRadius(float radius) {
this.radius = radius;
}
}
从输出结果可以看到,复制出来的大圣和原来的大圣拥有的金箍棒是一根,而不是每一个复制出来的大圣都有一根
下面使用深克隆改进上面的代码
深克隆实现
大圣本人:
public class TheGreatestSage {
private Monkey monkey = new Monkey();
public void change() throws IOException, ClassNotFoundException {
//创建大圣本尊对象
Monkey copyMonkey;
for (int i=0; i<2000; i++) {
copyMonkey = (Monkey) monkey.deepClone();
System.out.println("大圣本尊的生日:" + monkey.getBirthDate());
System.out.println("变化出来的大圣的生日:" + copyMonkey.getBirthDate());
System.out.println("大圣本尊== 变化出来的大圣:" + (monkey == copyMonkey));
System.out.println("大圣本尊的金箍棒== 变化出来的大圣的金箍棒:"
+ (monkey.getStaff() == copyMonkey.getStaff()));
}
}
public static void main(String[] args) throws IOException, ClassNotFoundException {
TheGreatestSage sage = new TheGreatestSage();
sage.change();
}
}
在Monkey类中增加深克隆方法:
public class Monkey implements Cloneable, Serializable {
/** 身高 */
private float height;
/** 体重 */
private float weight;
/** 金箍棒 */
private GoldRingedStaff staff;
/** 日期 */
private Date birthDate;
public Monkey() {
this.birthDate = new Date();
this.staff = new GoldRingedStaff();
}
/** 深拷贝方法 */
public Object deepClone() throws IOException, ClassNotFoundException {
//首先将对象写出到流
ByteArrayOutputStream bo = new ByteArrayOutputStream();
ObjectOutputStream oo = new ObjectOutputStream(bo);
oo.writeObject(this);
//然后将对象从流里读出来
ByteArrayInputStream bi = new ByteArrayInputStream(bo.toByteArray());
ObjectInputStream oi = new ObjectInputStream(bi);
return oi.readObject();
}
/** 浅拷贝方法 */
@Override
public Object clone() {
Monkey temp = null;
try {
temp = (Monkey) super.clone();
} catch (CloneNotSupportedException e) {
System.out.println("克隆失败!");
e.printStackTrace();
} finally {
return temp;
}
}
public float getHeight() {
return height;
}
public void setHeight(float height) {
this.height = height;
}
public float getWeight() {
return weight;
}
public void setWeight(float weight) {
this.weight = weight;
}
public Date getBirthDate() {
return birthDate;
}
public void setBirthDate(Date birthDate) {
this.birthDate = birthDate;
}
public GoldRingedStaff getStaff() {
return staff;
}
public void setStaff(GoldRingedStaff staff) {
this.staff = staff;
}
}
金箍棒类:
public class GoldRingedStaff implements Cloneable, Serializable {
/** 高度 */
private float height = 100.0f;
/** 半径 */
private float radius = 5.0f;
/** 构造函数 */
public GoldRingedStaff() {
}
/** 增长行为,每次调用高度和半径增加一倍 */
public void grow() {
this.height = this.height * 2;
this.radius = this.radius * 2;
}
/** 缩小行为,每次调用高度和半径减少一半 */
public void shrink() {
this.height = this.height / 2;
this.radius = this.radius / 2;
}
/** 移动 */
public void move() {
}
/** 高度取值方法 */
public float getHeight() {
return height;
}
/** 高度赋值方法 */
public void setHeight(float height) {
this.height = height;
}
/** 半径取值方法 */
public float getRadius() {
return radius;
}
/** 半径赋值方法 */
public void setRadius(float radius) {
this.radius = radius;
}
}
从输出中可以清楚的看到,大圣本人拥有的金箍棒和复制出来的金箍棒不是同一根了
孙悟空的身外身法术使用了Java设计模式:原型模式的更多相关文章
- 【设计模式】Java设计模式 - 原型模式
[设计模式]Java设计模式 - 原型模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 原创作品,更多关注我CSDN: 一个有梦有戏的人 准备将博客园.CSDN一起 ...
- 我的Java设计模式-原型模式
"不好意思,我是卧底!哇哈哈哈~"额......自从写了上一篇的观察者模式,就一直沉浸在这个角色当中,无法自拨.昨晚在看<使徒行者2>,有一集说到啊炮仗哥印钞票,我去, ...
- Java设计模式—原型模式
原型设计模式是一种比较简单的设计模式,在项目中使用的场景非常多. 个人理解: 原型模式实现了对Java中某个对象的克隆功能,即该对象的类必须implements实现Cloneable接口来标识为可被克 ...
- Java设计模式-原型模式(Prototype)
原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办法创建出更多同类型的对象.这就是选型模式的用意. 原型模式的结构 原型模式要求对象实现一个可以“克 ...
- java设计模式---原型模式
原型模式(Prototype):用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 原型模式结构图 通俗来说:原型模式就是深拷贝和浅拷贝的实现. 浅拷贝 只实现了值拷贝,对于引用对象还是 ...
- 4.java设计模式-原型模式(prototype)
在<JAVA与模式>一书中开头是这样描述原型(Prototype)模式的: 原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办法创建出更 ...
- Java设计模式原型模式
原型模式: – 通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式. – 就是java中的克隆技术,以某个对象为原型,复制出新的对象.显然,新的对象具备原型对象的特点 – 优势 ...
- java设计模式——原型模式
一. 定义与类型 定义:指原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象.不需要知道任何创建的细节,不调用构造函数 类型:创建型 二.使用场景 类初始化消耗较多资源 new 产生的一个对 ...
- Java设计模式——组合模式
JAVA 设计模式 组合模式 用途 组合模式 (Component) 将对象组合成树形结构以表示“部分-整体”的层次结构.组合模式使得用户对单个对象和组合对象的使用具有唯一性. 组合模式是一种结构型模 ...
随机推荐
- `curl -L` 解决 GitHub 的 raw.githubusercontent.com 无法连接问题
解决 GitHub 的 raw.githubusercontent.com 无法连接问题 在使用 curl 下载文件时,如果出现以下情况 curl: (7) Failed to connect to ...
- Linux流量查看工具
目录 监控总体带宽使用 nload.bmon.slurm.bwm-ng.cbm.speedometer和netload 监控总体带宽使用(批量式输出) vnstat.ifstat.dstat和coll ...
- 看完我的笔记不懂也会懂----javascript模块化
JavaScript模块化 模块化引子 模块化的历史进化 模块化规范 CommonJS规范 Node.js(服务器端) 下项目的结构分析 browerify(浏览器端) 下项目的结构分析 AMD规范 ...
- 基于CameraLink的逻辑综合和版图设计
前期接口设计用的是Vivado18.3+Modelsim10.6,逻辑综合及版图生成的环境是Ubuntu16,逻辑综合用的工具Design Compiler,生成版图用的工具是Encounter. 下 ...
- 使用 SVG transform rotate 解决画框中的数字跟随旋转的问题
问题描述 在图片上画框标注数字,旋转画布后,数字随之旋转,可读性不强,要求修改成无论画布怎么旋转,数字都是正向显示~ 原交互图示: 解决方案 先看下 dom 的结构 然后看下下面简单的代码 // 获取 ...
- ICPC题目选讲
Traveling in the grid world 题目描述 有一个 \(n\times m\) 的格点图,两点之间走他们的连线,但是这条连线不能恰好覆盖其他整点.还要求相邻两步之间的连线不能斜率 ...
- [笔记] 扩展Lucas定理
[笔记] 扩展\(Lucas\)定理 \(Lucas\)定理:\(\binom{n}{m} \equiv \binom{n/P}{m/P} \binom{n \% P}{m \% P}\pmod{P} ...
- Hibernate Validator异常HV000221解决办法
自建博客地址:https://www.bytelife.net,欢迎访问! 本文为博客同步发表文章,为了更好的阅读体验,建议您移步至我的博客 本文作者: Jeffrey 本文链接: https://w ...
- 2019 GDUT Rating Contest I : Problem C. Mooyo Mooyo
题面: C. Mooyo Mooyo Input file: standard input Output file: standard output Time limit: 1 second Memory ...
- Poj 3370
题目传送门:https://vjudge.net/problem/POJ-3370 题意:在n个数中找K个数使得他们的和为c的倍数. 题解:抽屉原理,同poj 2356 只不过写法上有所简化. 简化版 ...