结合JDK源码看设计模式——原型模式
定义:
指原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。不需要知道任何创建的细节,不调用构造函数
适用场景:
- 类初始化的时候消耗较多资源
- new产生的对象需要非常繁琐的过程
- 构造函数比较复杂
- 循环体中产生大量对象
详解:
接下来我们分下面几部分讲解:
- 原型模式的核心
- 深克隆和浅克隆
- JDK源码分析
1.原型模式的核心
其实很简单,就是实现Cloneable接口,然后重写clone()方法。上面我们已经说过 ,当你在上面的适用场景中的时候,按照我们平常的办法来说肯定是直接new对象出来,但是new对象特别多的时候就会消耗很多资源,并且效率也是比较缓慢的。所以我们引入原型模式的情况,其实我们只需要创建出一个原型来 ,剩下的完全可以通过克隆来达到创建新对象的目的。克隆是底层直接拿二进制流来克隆出新对象,然后对新对象进行特别的操作。
2.深克隆和浅克隆
这时候你心里可能会有疑惑,克隆不就克隆就行了吗?怎么还分浅克隆和深克隆。事实上在一个类中如果有另外的类的实例作为属性的话,正常使用Object.clone()方法,这个对象成员是无法被克隆的,也就是浅克隆。所以你怎么改原型中的对象成员,后面克隆的版本中这个对象成员就会一直跟原型一样。但是我们的目标是创建新对象来进行特定的操作,也就是希望每个对象里面的值不会跟他人共享。新对象是新对象,原型是原型。所以我们需要深克隆。下面我举几个例子来看看
public class Student implements Cloneable{
private int age;
private Date date;
private String name;
public void setAge(int age) {
this.age = age;
}
public void setDate(Date date) {
this.date = date;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Student{" +
"age=" + age +
", date=" + date +
", name='" + name + '\'' +
'}';
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
然后我们写一下测试类
class Test {
public static void main(String[] args) throws CloneNotSupportedException {
Student stu=new Student();
Student stu1= (Student) stu.clone();
Date date=new Date(0L);
int i=10;
stu.setDate(date);
stu1.setDate(date);
stu.setAge(i);
stu1.setAge(i);
System.out.println(stu);
System.out.println(stu1);
date.setTime(666666666L);
stu.setDate(date);
i=11;
stu.setAge(i);
System.out.println(stu);
System.out.println(stu1);
}
}
最终输出结果就是

分析一下

注意看这4行,我们上面是两个都调用了set方法,下面只有stu原型调用了set方法,但是最终却两个对象中的值一起改了。可能细心的的注意到Age只有stu改了,这是因为int类型基本数据类型。而Date类型的对象成员就不行了,实际上只有Student这个类进行了克隆,但是Student里面的对象成员变量没有进行克隆。所以那个对象还是那个对象。

上面就是浅克隆。要想做到深克隆,可以在clone方法里面clone出对应的对象成员结果及代码如下


上面我重写了clone()方法,实现了深克隆
3.JDK源码解析
其实我们主要理解拷贝原型来创建新的对象,拷贝是比new更快的一个创建对象的方法,当你需要大批量创建新对象而且都是同一个类的对象的时候可以考虑用原型模式。但是千万千万注意就是一般的克隆只是浅克隆(浅克隆:只是对象的hash值不一样,但是对象里面的对象成员变量的hash值是一样的)有些场景可能是需要我们深克隆的,这时候就需要我们重写Object.clone()方法。就拿ArrayList中的clone()方法来看
public Object clone() {
try {
ArrayList<?> v = (ArrayList<?>) super.clone();//先克隆出一个ArrayList
v.elementData = Arrays.copyOf(elementData, size);//将原型中的数据拷贝到新的ArrayList中
v.modCount = 0;//把修改次数改为0
return v;
} catch (CloneNotSupportedException e) {
// this shouldn't happen, since we are Cloneable
throw new InternalError(e);
}
}
相信看懂了上面的实例,你也能理解ArrayList中这段代码到底是做什么
总结:
尽管我们会用了原型,知道拷贝比new快,知道深克隆,但是具体的还是看业务场景的需求,希望能多理解适用场景的那几种情况。
结合JDK源码看设计模式——原型模式的更多相关文章
- 结合JDK源码看设计模式——桥接模式
前言: 在我们还没学习框架之前,肯定都学过JDBC.百度百科对JDBC是这样介绍的[JDBC(Java DataBase Connectivity,java数据库连接)是一种用于执行SQL语句的Jav ...
- 结合JDK源码看设计模式——模板方法模式
前言: 相信很多人都听过一个问题:把大象关进冰箱门,需要几步? 第一,把冰箱门打开:第二,把大象放进去:第三,把冰箱门关上.我们可以看见,这个问题的答案回答的很有步骤.接下来我们介绍一种设计模式--模 ...
- 结合JDK源码看设计模式——迭代器模式
前言: Iterator翻译过来就是迭代器的意思.在前面的工厂模式中就介绍过了iterator,不过当时介绍的是方法,现在从Iterator接口的设计来看,似乎又是一种设计模式,下面我们就来讲讲迭代器 ...
- 结合JDK源码看设计模式——建造者模式
概念: 将一个复杂对象的构建与它的表示分离.使得同样构建过程可以创建不同表示适用场景: 一个对象有很多属性的情况下 想把复杂的对象创建和使用分离 优点: 封装性好,扩展性好 详解: 工厂模式注重把这个 ...
- 结合JDK源码看设计模式——策略模式
前言: 现在电商已经成为我们生活中不可或缺的购物渠道,同时各大商家会针对不同的时间做出不同的折扣,这在我们看来就是一种营销手段,也是一种策略,今天我们就来讲讲JDK中的策略模式是怎么样的. 一.定义 ...
- 结合JDK源码看设计模式——组合模式
前言: 相信大家都打开过层级很多很多的文件夹.如果把第一个文件夹看作是树的根节点的话,下面的子文件夹就可以看作一个子节点.不过最终我们寻找的还是文件夹中的文件,文件可以看做是叶子节点.下面我们介绍一种 ...
- 结合JDK源码看设计模式——单例模式
定义: 保证一个类仅有一个实例,并提供一个全局访问点 适用场景: 确保任何情况下这个对象只有一个实例 详解: 私有构造器 单利模式中的线程安全+延时加载 序列化和反序列化安全, 防止反射攻击 结合JD ...
- 结合JDK源码看设计模式——简单工厂、工厂方法、抽象工厂
三种工厂模式的详解: 简单工厂模式: 适用场景:工厂类负责创建的对象较少,客户端只关心传入工厂类的参数,对于如何创建对象的逻辑不关心 缺点:如果要新加产品,就需要修改工厂类的判断逻辑,违背软件设计中的 ...
- 结合JDK源码看设计模式——享元模式
前言 在说享元模式之前,你一定见到过这样的面试题 public class Test { public static void main(String[] args) { Integer a=Inte ...
随机推荐
- HBuilder 打包流程
1.运行HBuilder---百度搜索HBuilder,官网下载安装包,解压,运行HBuilder.exe.注册账号,并登陆 2.新建app---在左边右键,选择新建APP,或者,点击中间的新建app ...
- springMVC(spring)+WebSocket案例(获取请求参数)
开发环境(最低版本):spring 4.0+java7+tomcat7.0.47+sockjs 前端页面要引入: <script src="http://cdn.jsdelivr.ne ...
- Netty中的连接管理
连接管理是我们首先需要关注的,检测空闲连接以及超时对于及时释放资源来说是至关重要的.由于这是一项常见的任务,Netty特地为它提供了几个ChannelHandler实现. 用于空闲连接以及超时的Cha ...
- 深夜学算法之SkipList:让链表飞
1. 前言 上次写Python操作LevelDB时提到过,有机会要实现下SkipList.摘录下wiki介绍: 跳跃列表是一种随机化数据结构,基于并联的链表,其效率可比拟二叉查找树. 我们知道对于有序 ...
- Java开发岗面试知识点解析
本文作者参加过多场面试,应聘岗位均为 Java 开发方向.在不断的面试中,分类总结了 Java 开发岗位面试中的一些知识点. 主要包括以下几个部分: Java 基础知识点 Java 常见集合 高并发编 ...
- Debian9桌面设置
本文由荒原之梦原创,原文链接:http://zhaokaifeng.com/?p=665 新安装的Debian9桌面上啥都没有,就像这样: 图 1 虽然很简洁,但是用着不是很方便,下面我们就通过一些设 ...
- Sublime Text 3安装SFTP插件
前言: 最近在学习网页设计,陆续接触到了HTML.CSS和JavaScript,写的代码越来越多了,也越来越感觉到将代码上传到服务器上的流程太繁琐了.一开始我是用虚拟主机提供的控制面板下载上传网页代码 ...
- 解决持久化数据太大,单个节点的硬盘无法存储的问题;解决运算量太大,单个节点的内存、CPU无法处理的问题
需要学习的技术很多,要自学新知识也不是一件容易的事,选择一个自己比较感兴趣的会是一个比较好的开端,于是,打算学一学分布式系统. 带着问题,有目的的学习,先了解整体架构,在深入感兴趣的细节,这是我的计划 ...
- redis与python交互
import redis #连接 r=redis.StrictRedis(host="localhost",port=6379,password="sunck" ...
- nodejs版本更新问题:express不是内部或外部命令
版本更新后,我们使用熟悉的npm install -g express命令安装,但是,安装成功之后居然提示express不是内部或外部命令. nodejs小问题:[1]express不是内部或外部命令 ...