java设计模式——原型模式
一. 定义与类型
定义:指原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。不需要知道任何创建的细节,不调用构造函数
类型:创建型
二.使用场景
类初始化消耗较多资源
new 产生的一个对象需要非常繁琐的过程(数据准备,访问权限等)
构造函数比较复杂
循环体中生产大量对象时
三.优缺点
优点:
原型模式性能比直接new一个对象性能高,简化创建过程
缺点:
必须配备克隆方法,
对克隆复杂对象或对克隆出的对象进行复杂改造时,容易引入风险
深拷贝,浅拷贝要运用得当
四. 扩展
深克隆:对于引用类型,如果需要指向不同的对象,而对于某个对象的引用类型的时候,必须要显式的去写对那个属性进行深克隆
浅克隆:
五. Coding
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:11
**/
public class Mail implements Cloneable {
private String name;
private String emailAddress;
private String content;
public Mail() {
System.out.println("Mail Class Constructor");
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getEmailAddress() {
return emailAddress;
} public void setEmailAddress(String emailAddress) {
this.emailAddress = emailAddress;
} public String getContent() {
return content;
} public void setContent(String content) {
this.content = content;
} @Override
public String toString() {
return "Mail{" +
"name='" + name + '\'' +
", emailAddress='" + emailAddress + '\'' +
", content='" + content + '\'' + super.toString() +
'}';
} @Override
protected Object clone() throws CloneNotSupportedException {
System.out.println("clone mail object");
return super.clone();
}
}
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:17
**/
public class MailUtil {
public static void sendMail(Mail mail) {
String outputContent = "向{0}同学,邮件地址:{1},邮件内容:{2}发送成功";
System.out.println(MessageFormat.format(outputContent,mail.getName(),mail.getEmailAddress(),mail.getContent()));
} public static void saveOriginMailRecord(Mail mail) {
System.out.println("存储originMail记录,originMail:" + mail.getContent());
}
}
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:20
**/
public class Test {
public static void main(String[] args) throws CloneNotSupportedException {
Mail mail = new Mail();
mail.setContent("初始化模板");
System.out.println("初始化mail:" + mail);
for (int i = 0; i < 10; i++) {
//克隆的时候,并不会使用原对象的构造器
Mail mailTemp = (Mail) mail.clone();
mailTemp.setName("姓名" + i);
mailTemp.setEmailAddress("姓名" + i + "@qq.com");
mailTemp.setContent("恭喜您,中奖了");
MailUtil.sendMail(mailTemp);
}
MailUtil.saveOriginMailRecord(mail);
}
}
从上面的代码不难看出,其实原型模式就是实现了cloneable接口,在创建一个不同的对象,来完成邮件的发送,而保留了原来的邮件模板。由于clone不会,调用原对象的构造器,所以在效率上比直接new 对象要高。但是,因为在Mail类中的属性都是简单类型,所以在clone的时候,基本上不会出现上面问题。但是看下面一个实体:
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:33
**/
public class Pig implements Cloneable {
private String name;
private Date birthDay; public Pig(String name, Date birthDay) {
this.name = name;
this.birthDay = birthDay;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Date getBirthDay() {
return birthDay;
} public void setBirthDay(Date birthDay) {
this.birthDay = birthDay;
} @Override
public String toString() {
return "Pig{" +
"name='" + name + '\'' +
", birthDay=" + birthDay +
'}';
} @Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:34
**/
public class Test {
public static void main(String[] args) throws CloneNotSupportedException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
Date birthDay = new Date(0L);
Pig pig = new Pig("佩奇", birthDay);
Pig pig1 = (Pig) pig.clone();
System.out.println(pig);
System.out.println(pig1); pig.getBirthDay().setTime(6666666666666L);
/**
* 通过上面对birthDay进行操作,如果是浅拷贝,那么pig与pig1的date对象都会改变:
* 因为在浅拷贝的时候,两个对象中的引用对象date,都是引用同一个对象,所以改变了一个,那么两个都会改变
*
* 如果是深拷贝,那么pig与pig1引用的date对象就是不一样的,改变其中一个,对另一个并没有影响。
*
* 由于深克隆,浅克隆的关系,也算是原型模式的一个坑。(原则是,都会使用深克隆,不然就算是给项目埋坑)
*
*/
System.out.println(pig);
System.out.println(pig1);
}
}
输出结果为:

上面的注释和结果,其实也清楚的看到了。我只改变了pig对象中的birthDay,但是pig1中也改变了。如果clone的对象中存在引用类型的对象,那么如果是浅拷贝,拷贝与被拷贝出的对象的引用对象都是指向同一地址的,所以改变其中一个,另一个也会改变。这时候,如果根据需求就必须使用深拷贝。
也就是对于对象中的引用对象也要进行clone。看下面代码:
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:33
**/
public class Pig implements Cloneable {
private String name;
private Date birthDay; public Pig(String name, Date birthDay) {
this.name = name;
this.birthDay = birthDay;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Date getBirthDay() {
return birthDay;
} public void setBirthDay(Date birthDay) {
this.birthDay = birthDay;
} @Override
public String toString() {
return "Pig{" +
"name='" + name + '\'' +
", birthDay=" + birthDay +
'}';
} @Override
protected Object clone() throws CloneNotSupportedException {
Pig pig = (Pig) super.clone(); //深克隆
pig.birthDay = (Date) pig.birthDay.clone();
return pig;
}
}
输出结果:

可以看到,Pig中重写了clone方法,对Date对象也进行了clone,从而使得,引用对象指向不同的地址。所以改变pig中的birthDay,对于pig1并没有影响。
由于clone方法,可以用原型模式拷贝来破坏单例模式:
/**
* @program: designModel
* @description:
* @author: YuKai Fan
* @create: 2018-12-13 16:34
**/
public class Test {
public static void main(String[] args) throws CloneNotSupportedException, NoSuchMethodException, InvocationTargetException, IllegalAccessException { /**
* 使用原型模式,克隆破坏单例模式
*
* 这种情况的解决方式:
* 要么单例模式类不去实现Cloneable接口,要么就重写clone方法,直接返回getInstance()方法,这个对象的实例
*/
HungrySingleton hungrySingleton = HungrySingleton.getInstance();
Method method = hungrySingleton.getClass().getDeclaredMethod("clone");
method.setAccessible(true);
HungrySingleton cloneHungrySingleton = (HungrySingleton) method.invoke(hungrySingleton);
System.out.println(hungrySingleton);
System.out.println(cloneHungrySingleton);
}
}
六. 源码分析
基本上只要知道了哪些类使用了Cloneable就知道,原型模式如何使用。
比如ArrayList,HashMap类都重写了clone方法
Mybatis中的CacheKey类也重写了clone()方法
java设计模式——原型模式的更多相关文章
- 【设计模式】Java设计模式 - 原型模式
[设计模式]Java设计模式 - 原型模式 不断学习才是王道 继续踏上学习之路,学之分享笔记 总有一天我也能像各位大佬一样 原创作品,更多关注我CSDN: 一个有梦有戏的人 准备将博客园.CSDN一起 ...
- 我的Java设计模式-原型模式
"不好意思,我是卧底!哇哈哈哈~"额......自从写了上一篇的观察者模式,就一直沉浸在这个角色当中,无法自拨.昨晚在看<使徒行者2>,有一集说到啊炮仗哥印钞票,我去, ...
- Java设计模式—原型模式
原型设计模式是一种比较简单的设计模式,在项目中使用的场景非常多. 个人理解: 原型模式实现了对Java中某个对象的克隆功能,即该对象的类必须implements实现Cloneable接口来标识为可被克 ...
- Java设计模式-原型模式(Prototype)
原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办法创建出更多同类型的对象.这就是选型模式的用意. 原型模式的结构 原型模式要求对象实现一个可以“克 ...
- java设计模式---原型模式
原型模式(Prototype):用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 原型模式结构图 通俗来说:原型模式就是深拷贝和浅拷贝的实现. 浅拷贝 只实现了值拷贝,对于引用对象还是 ...
- 4.java设计模式-原型模式(prototype)
在<JAVA与模式>一书中开头是这样描述原型(Prototype)模式的: 原型模式属于对象的创建模式.通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办法创建出更 ...
- Java设计模式原型模式
原型模式: – 通过new产生一个对象需要非常繁琐的数据准备或访问权限,则可以使用原型模式. – 就是java中的克隆技术,以某个对象为原型,复制出新的对象.显然,新的对象具备原型对象的特点 – 优势 ...
- PHP 设计模式 原型模式(Prototype)之深/浅拷贝
看PHP 设计模式 原型模式(Prototype)时,衍生出一个扩展问题之 原型拷贝的浅拷贝和深拷贝问题(不管写Java还是写PHP还是写JS时都多多少少遇到过对象拷贝问题) 比如写前端页面时 ...
- 10. 星际争霸之php设计模式--原型模式
题记==============================================================================本php设计模式专辑来源于博客(jymo ...
随机推荐
- dorado 常用
如果要设置模糊查询, 一般要在QueryCommand中这样写: var name = dsQuery.getValue("NAME"); var parameters = com ...
- jmeter小问题解决方案合集
问题1.在http请求,post的body中输入中文,显示乱码,怎么解决? 在jmeter的bin目录下,找到这个文件jmeter.properties,把jsyntaxtextarea.font.f ...
- Javascript专题(三)b.各种轮播和细节分析--上下滚动轮播
这一次,我们用原生JS实现上下滚动方式的轮播.顺带学习一下用JS来创建HTML元素. 上一次写的轮播是淡入淡出效果的,相对来说其实是比较简单的. github源码: 上下轮播源码-github A. ...
- HQL和SQL
hql是面向对象查询,格式:from + 类名 + 类对象 + where + 对象的属性 sql是面向数据库表查询,格式:from + 表名 + where + 表中字段 1.查询 一般在hiber ...
- C++之Vect
在C++中数组和向量都是多同类元素的集合,他们也有很明显的区别 1 数组属于静态分配,编译之前必须知道数组的大小,一旦确定就不能更改:2个数组之间不能直接赋值实现拷贝,而必须显式用for或者拷贝函数拷 ...
- 深入理解C#中的IDisposable接口(转)
转自:https://www.cnblogs.com/wyt007/p/9304564.html 写在前面 在开始之前,我们需要明确什么是C#(或者说.NET)中的资源,打码的时候我们经常说释放资源, ...
- IIS错误HTTP 错误 500.21 - Internal Server Error
原因:在安装Framework v4.0之后,再启用IIS,导致Framework没有完全安装 解决:以管理员身份运行cmd->输入“%windir%\Microsoft.NET\Framewo ...
- oop典型应用:实体类
1. 要知道这个图三者的关系 2.实体类属性类型与数据库类型 3.readonly与const的对比 两者的区别如下: ①const能修饰类中的字段(field)或者局部变量(local variab ...
- 《Head First 设计模式》之单件模式
单件模式(Singleton) ——确保一个类只有一个实例,并提供全局访问点. 有一些对象其实我们只需要一个,比如线程池.缓存.对话框.处理偏好设置和注册表的对象.日志对象.如果制造出多个实例,就会导 ...
- 移植mavlink协议到STM32详细教程
1准备材料, 首先准备一个带串口的stm32程序(这里选用整点原子的官方串口例程这里自己去找不讲)作者:恒久力行 QQ:624668529,然后去mavlink官网下载mavlink源码,这里重点讲解 ...