GOF23设计模式之原型模式(prototype)
一、原型模式概述
1.通过new产生一个对象需要非常繁琐的数据准备和访问权限,则可以使用原型模式。
2.就是java中的克隆技术,以某个对象为原型,复制出新的对象,显然,新的对象具备原型对象的特点。
3.优势:效率高(直接克隆,避免了重新执行构造函数的过程)。
4.克隆类似于new,但是不同于new。
new创建出来的对象属性采用默认值,克隆出来的对象属性值完全和原型对象相同,并且克隆出的新对象不会影响原型对象,然后,再修改克隆对象的值。
是否使用原型模式比较:
1.如果需要创建大量不费时的对象,new的对象和clone的对象效率相当
2.如果需要创建大量耗时的对象,建议使用原型模式
二、原型模式三种实现方式
1.浅复制:类中实现 Cloneable接口,重写clone方法
2.深复制:类中实现 Cloneable接口,重写clone方法时将对象一起克隆
3.深复制:类中实现 Serializable接口,利用反序列化实现深克隆
三、使用浅复制实现原型模式
1.使用原型模式克隆多利羊
/**
* 原型模式(浅克隆)
* @author CL
*
*/
public class Sheep implements Cloneable {
private String name; public Object clone() throws CloneNotSupportedException {
return super.clone();
} public Sheep() {
} public Sheep(String name) {
this.name = name;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} }
2.测试
public class TestSheep {
public static void main(String[] args) throws Exception {
Sheep s1 = new Sheep("多利");
Sheep s2 = (Sheep) s1.clone(); //克隆对象
System.out.println(s1.hashCode()+"(原型对象)-->"+s1.getName());
System.out.println(s2.hashCode()+"(克隆对象)-->"+s2.getName());
s2.setName("少利"); //对克隆对象修改,不影响原对象的属性值
System.out.println("\n--------修改克隆对象的属性值后---------\n");
System.out.println(s1.hashCode()+"(原型对象)-->"+s1.getName());
System.out.println(s2.hashCode()+"(克隆对象)-->"+s2.getName());
}
}
控制台输出:
366712642(原型对象)-->多利
1829164700(克隆对象)-->多利 --------修改克隆对象的属性值后--------- 366712642(原型对象)-->多利
1829164700(克隆对象)-->少利
四、使用深复制实现原型模式(1)
1.使用原型模式克隆多利羊
import java.util.Date; /**
* 原型模式(深克隆)
* @author CL
*
*/
public class Sheep implements Cloneable {
private String name; private Birthday birthday; public Object clone() throws CloneNotSupportedException {
Sheep s = (Sheep) super.clone(); //克隆对象
s.birthday = (Birthday) this.birthday.clone();
return s;
} public Sheep() {
} public Sheep(String name, Birthday birthday) {
this.name = name;
this.birthday = birthday;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Birthday getBirthday() {
return birthday;
} public void setBirthday(Birthday birthday) {
this.birthday = birthday;
} } class Birthday implements Cloneable {
private Date birthday; protected Object clone() throws CloneNotSupportedException {
return super.clone();
} public Birthday() {
} public Birthday(Date birthday) {
this.birthday = birthday;
} public Date getBirthday() {
return birthday;
} public void setBirthday(Date birthday) {
this.birthday = birthday;
}
}
2.测试
import java.util.Date; /**
* 测试原型模式(深克隆)
* @author CL
*
*/
public class TestSheep { public static void main(String[] args) throws Exception {
Birthday date = new Birthday(new Date(5456464L));
Sheep s1 = new Sheep("多利", date);
Sheep s2 = (Sheep) s1.clone(); //克隆对象 System.out.println(s1.hashCode()+"(原型对象)-->"+s1.getName()+"-->"+s1.getBirthday().getBirthday());
System.out.println(s2.hashCode()+"(克隆对象)-->"+s2.getName()+"-->"+s2.getBirthday().getBirthday()); date.setBirthday(new Date()); //对原对象修改,克隆对象的属性值不改变 System.out.println("\n--------修改克隆对象的属性值后---------\n"); System.out.println(s1.hashCode()+"(原型对象)-->"+s1.getName()+"-->"+s1.getBirthday().getBirthday());
System.out.println(s2.hashCode()+"(克隆对象)-->"+s2.getName()+"-->"+s2.getBirthday().getBirthday());
} }
控制台输出:
366712642(原型对象)-->多利-->Thu Jan 01 09:30:56 CST 1970
1550089733(克隆对象)-->多利-->Thu Jan 01 09:30:56 CST 1970
366712642(原型对象)-->多利-->Fri Dec 29 17:03:26 CST 2017
1550089733(克隆对象)-->多利-->Thu Jan 01 09:30:56 CST 1970
五、使用深复制实现原型模式(2)
1.使用原型模式克隆多利羊
import java.io.Serializable;
import java.util.Date; /**
* 原型模式(利用反序列化实现深克隆)
* @author CL
*
*/
public class Sheep implements Serializable {
private String name; private Birthday birthday; public Sheep() {
} public Sheep(String name, Birthday birthday) {
this.name = name;
this.birthday = birthday;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Birthday getBirthday() {
return birthday;
} public void setBirthday(Birthday birthday) {
this.birthday = birthday;
} } class Birthday implements Serializable {
private Date birthday; public Birthday() {
} public Birthday(Date birthday) {
this.birthday = birthday;
} public Date getBirthday() {
return birthday;
} public void setBirthday(Date birthday) {
this.birthday = birthday;
}
}
2.测试
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.util.Date; /**
* 测试原型模式(利用反序列化实现深克隆)
* @author CL
*
*/
public class TestSheep { public static void main(String[] args) throws Exception {
Birthday date = new Birthday(new Date(5456464L));
Sheep s1 = new Sheep("多利", date); //利用反序列化实现深克隆
//1. 序列化
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(s1);
byte[] bytes = bos.toByteArray(); //2. 反序列化
ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
ObjectInputStream ois = new ObjectInputStream(bis);
Sheep s2 = (Sheep) ois.readObject(); System.out.println(s1.hashCode()+"(原型对象)-->"+s1.getName()+"-->"+s1.getBirthday().getBirthday());
System.out.println(s2.hashCode()+"(克隆对象)-->"+s2.getName()+"-->"+s2.getBirthday().getBirthday()); date.setBirthday(new Date()); //对原对象修改,克隆对象的属性值不改变 System.out.println("\n--------------------修改克隆对象的属性值后---------------------\n"); System.out.println(s1.hashCode()+"(原型对象)-->"+s1.getName()+"-->"+s1.getBirthday().getBirthday());
System.out.println(s2.hashCode()+"(克隆对象)-->"+s2.getName()+"-->"+s2.getBirthday().getBirthday());
} }
控制台输出:
1442407170(原型对象)-->多利-->Thu Jan 01 09:30:56 CST 1970
1173230247(克隆对象)-->多利-->Thu Jan 01 09:30:56 CST 1970 --------------------修改克隆对象的属性值后--------------------- 1442407170(原型对象)-->多利-->Fri Dec 29 17:11:21 CST 2017
1173230247(克隆对象)-->多利-->Thu Jan 01 09:30:56 CST 1970
六、原型模式常见应用场景
1.原型模式很少单独出现,一般是和工厂方法模式一起出现。通过clone的方法创建一个对象,然后由工厂方法提供给调用者;
2.Spring中的bean的创建其实就是两种:单例模式和原型模式(+工厂模式);
3.………………
//对原型对象的修改
GOF23设计模式之原型模式(prototype)的更多相关文章
- GOF23设计模式之原型模式
GOF23设计模式之原型模式 1)通过 new 产生一个对象需要飞船繁琐的数据准备或访问权限,则可以使用原型模式. 2)就算 java 中的克隆技术,以某个对象为原型,复制出新的对象.显然,新的对象具 ...
- 乐在其中设计模式(C#) - 原型模式(Prototype Pattern)
原文:乐在其中设计模式(C#) - 原型模式(Prototype Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 原型模式(Prototype Pattern) 作者:weba ...
- 二十四种设计模式:原型模式(Prototype Pattern)
原型模式(Prototype Pattern) 介绍用原型实例指定创建对象的种类,并且通过拷贝这个原型来创建新的对象.示例有一个Message实体类,现在要克隆它. MessageModel usin ...
- 【GOF23设计模式】原型模式
来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_原型模式.prototype.浅复制.深复制.Cloneable接口 浅复制 package com.test.prot ...
- [设计模式] 4 原型模式 prototype
设计模式:可复用面向对象软件的基础>(DP)本文介绍原型模式和模板方法模式的实现.首先介绍原型模式,然后引出模板方法模式. DP书上的定义为:用原型实例指定创建对象的种类,并且通过拷贝这些原型创 ...
- 设计模式 笔记 原型模式 prototype
//---------------------------15/04/07---------------------------- //prototype 原型模式--对象创建型模式 /* 1:意图: ...
- python 设计模式之原型模式 Prototype Pattern
#引入 例子1: 孙悟空拔下一嘬猴毛,轻轻一吹就会变出好多的孙悟空来. 例子2:寄个快递下面是一个邮寄快递的场景:“给我寄个快递.”顾客说.“寄往什么地方?寄给……?”你问.“和上次差不多一样,只是邮 ...
- 【UE4 设计模式】原型模式 Prototype Pattern
概述 描述 使用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象.如孙悟空猴毛分身.鸣人影之分身.剑光分化.无限剑制 原型模式是一种创建型设计模式,允许一个对象再创建另外一个可定制的对象, ...
- 【设计模式】—— 原型模式Prototype
前言:[模式总览]——————————by xingoo 模式意图 由于有些时候,需要在运行时指定对象时哪个类的实例,此时用工厂模式就有些力不从心了.通过原型模式就可以通过拷贝函数clone一个原有的 ...
随机推荐
- firefox与ie 的javascript区别
1. Document.form.item 问题 现有问题: 现有代码中存在许多 document.formName.item("itemName") 这样的语句,不能在 ...
- uname命令行
常用命令uname -v # uname -i #uname -a dream361@master:~$ uname -n #主机名称 master dream361@master:~$ uname ...
- 【Seajs源码分析】3. 工具方法2
util-request.js 动态加载模块 /** * util-request.js - The utilities for requesting script and style files * ...
- DRF中两大视图基类APIView/GenericAPIView的介绍
(1)APIView rest_framework.views.APIView APIView是REST framework提供的所有视图的基类,继承自Django的View父类. APIView与V ...
- Ubuntu12.04 中文输入法设置
1.ibus输入法 Ubuntu系统安装后已经自带了ibus输入法,在英语环境下默认不启动. 配置ibus自动启动可 以在ubuntu系统菜单上选择System(系统)--- Preferences( ...
- jquery表单验证插件 jquery.form.js-转
来自:http://www.cnblogs.com/luluping/archive/2009/04/15/1436177.html Form插件,支持Ajax,支持Ajax文件上传,功能强大,基本满 ...
- 《Drools7.0.0.Final规则引擎教程》第4章 4.3 日历
日历 日历可以单独应用于规则中,也可以和timer结合使用在规则中使用.通过属性calendars来定义日历.如果是多个日历,则不同日历之间用逗号进行分割. 在Drools中,日历的概念只是将日历属性 ...
- CUDA Samples: Julia
以下CUDA sample是分别用C++和CUDA实现的绘制Julia集曲线,并对其中使用到的CUDA函数进行了解说,code参考了<GPU高性能编程CUDA实战>一书的第四章,各个文件内 ...
- Vim技能修炼教程(15) - 时间和日期相关函数
Vimscript武器库 前面我们走马观花地将Vimscript的大致语法过了一遍.下面我们开始深入看一下Vimscript都给我们准备了哪些武器.如果只用这些武器就够了,那么就太好了,只用Vimsc ...
- Android Studio真机测试失败-----''No target device found" (转)
参考文章: https://blog.csdn.net/chang_sir/article/details/51755572 今天想用真机测试一个程序,却报出这样一个Error"No tar ...