java设计模式4——原型模式
java设计模式4——原型模式
1、写在前面
本节内容与C++语言的复制构造函数、浅拷贝、深拷贝极为相似,因此建议学习者可以先了解C++的该部分的相关知识,或者学习完本节内容后,也去了解C++的相应内容,进行比对学习。
2、原型模式介绍
原型模式(Prototype Pattern)是用于创建重复的对象,同时又能保证性能。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
这种模式是实现了一个原型接口,该接口用于创建当前对象的克隆。当直接创建对象的代价比较大时,则采用这种模式。例如,一个对象需要在一个高代价的数据库操作之后被创建。我们可以缓存该对象,在下一个请求时返回它的克隆,在需要的时候更新数据库,以此来减少数据库调用。
3、java实现克隆的核心
1、实现一个接口(Cloneable)
2、重写一个方法(clone())
clone()方法的源码分析
protected native Object clone() throws CloneNotSupportedException;
由方法声明的放回值类型=>native,可知该方法实际上是一个C++封装好的方法,由java来进行调用,相当于C++语言的复制构造函数,但是又有所区别。
4、第一种原型模式实现(浅拷贝)
4.1、建立视频的原型类
package com.xgp.company.创建型模式.第四种_原型模式.demo1;
import java.util.Date;
/**
* 1、实现一个接口
* 2、重写一个方法
*/
//视频原型类
public class Video implements Cloneable {
private String name;
private Date createTime;
/**
* 重写克隆方法
* @return
* @throws CloneNotSupportedException
*/
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
public Video() {
}
public Video(String name, Date createTime) {
this.name = name;
this.createTime = createTime;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Date getCreateTime() {
return createTime;
}
public void setCreateTime(Date createTime) {
this.createTime = createTime;
}
@Override
public String toString() {
return "Video{" +
"name='" + name + '\'' +
", createTime=" + createTime +
'}';
}
}
4.2、建立复制的客户端类
package com.xgp.company.创建型模式.第四种_原型模式.demo1;
import java.util.Date;
/**
* 客户端:克隆
*/
public class Client {
public static void main(String[] args) throws CloneNotSupportedException {
//原型对象
Video v1 = new Video("狂神说java", new Date());
System.out.println("v1 = " + v1);
System.out.println("v1 = " + v1.hashCode());
System.out.println("========================");
//v1 克隆 v2
Video v2 = (Video) v1.clone();
System.out.println("v2 = " + v2);
System.out.println("v2 = " + v2.hashCode());
}
}
运行结果:
v1 = Video{name='狂神说java', createTime=Fri Feb 14 10:26:57 CST 2020}
v1 = 1836019240
========================
v2 = Video{name='狂神说java', createTime=Fri Feb 14 10:26:57 CST 2020}
v2 = 325040804
5、弊端
5.1、揭露弊端的代码
/**
* 演示浅克隆弊端
* @param args
* @throws CloneNotSupportedException
*/
public static void main(String[] args) throws CloneNotSupportedException {
//原型对象
Date date = new Date();
Video v1 = new Video("狂神说java", date);
System.out.println("v1 = " + v1);
System.out.println("v1 = " + v1.hashCode());
//v1 克隆 v2
Video v2 = (Video) v1.clone();
System.out.println("v2 = " + v2);
System.out.println("v2 = " + v2.hashCode());
System.out.println("========================");
date.setTime(22222222);
System.out.println("v1 = " + v1);
System.out.println("v1 = " + v1.hashCode());
//v1 克隆 v2
System.out.println("v2 = " + v2);
System.out.println("v2 = " + v2.hashCode());
}
5.2、弊端代码的运行结果
v1 = Video{name='狂神说java', createTime=Fri Feb 14 10:29:02 CST 2020}
v1 = 1836019240
v2 = Video{name='狂神说java', createTime=Fri Feb 14 10:29:02 CST 2020}
v2 = 325040804
========================
v1 = Video{name='狂神说java', createTime=Thu Jan 01 14:10:22 CST 1970}
v1 = 1836019240
v2 = Video{name='狂神说java', createTime=Thu Jan 01 14:10:22 CST 1970}
v2 = 325040804
5.3、弊端分析
从运行结果可以看出,当改变一个被v1,v2都引用的时间对象时,两者都发生了改变,没有将引用的对象进行复制,因此称之为浅拷贝。
5.4、模型图如下:
通过上面的模型图可以看到,被引用的对象还是只有一份,因此存在很大的弊端。在C++中,如果存在这种引用关闭,当程序员两次删除对象时,会出现第二次删对象时发生异常的情况。虽然在java语言中,有着gc机制,不需要程序员手动的去释放对象,不会出现重复删除的错误。但是,当被引用的对象发生改变时,很有可能会发生数据上的不正确。
6、改进(深拷贝模式)
6.1、我们期望想要改进后的模型图如下:被引用的对象也复制了一份,互不干扰。
6.2、实现方式:改造重写的克隆方法
@Override
protected Object clone() throws CloneNotSupportedException {
/**
* 改造克隆方法,进行生克隆
*/
Object obj = super.clone();
Video v = (Video) obj;
//将对象的属性进行克隆
v.createTime = (Date) this.createTime.clone();
return obj;
}
6.3、此时的运行结果为:
v1 = Video{name='狂神说java', createTime=Fri Feb 14 10:37:33 CST 2020}
v1 = 1836019240
v2 = Video{name='狂神说java', createTime=Fri Feb 14 10:37:33 CST 2020}
v2 = 325040804
========================
v1 = Video{name='狂神说java', createTime=Thu Jan 01 14:10:22 CST 1970}
v1 = 1836019240
v2 = Video{name='狂神说java', createTime=Fri Feb 14 10:37:33 CST 2020}
v2 = 325040804
6.4、当原型引用的对像改变时,复制的对象并不发生改变,被引用的对象被复制了两份,因此称之为深拷贝。
java设计模式4——原型模式的更多相关文章
- JAVA 设计模式之原型模式
目录 JAVA 设计模式之原型模式 简介 Java实现 1.浅拷贝 2.深拷贝 优缺点说明 1.优点 2.缺点 JAVA 设计模式之原型模式 简介 原型模式是六种创建型设计模式之一,主要应用于创建相同 ...
- java设计模式之原型模式
原型模式概念 该模式的思想就是将一个对象作为原型,对其进行复制.克隆,产生一个和原对象类似的新对象.java中复制通过clone()实现的.clone中涉及深.浅复制.深.浅复制的概念如下: ⑴浅复制 ...
- java设计模式之五原型模式(Prototype)
原型模式虽然是创建型的模式,但是与工程模式没有关系,从名字即可看出,该模式的思想就是将一个对象作为原型,对其进行复制.克隆,产生一个和原对象类似的新对象.本小结会通过对象的复制,进行讲解.在Java中 ...
- JAVA设计模式之 原型模式【Prototype Pattern】
一.概述: 使用原型实例指定创建对象的种类,而且通过拷贝这些原型创建新的对象. 简单的说就是对象的拷贝生成新的对象(对象的克隆),原型模式是一种对象创建型模式. 二.使用场景: 创建新的对象能够通过对 ...
- JAVA 设计模式之 原型模式详解
原型模式(Prototype Pattern)是指原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 原型模式利用的是克隆的原理,创建新的对象,JDK提供的Cloneable 和JSON. ...
- JAVA设计模式之原型模式(prototype)
原型模式: 原型模式又叫克隆模式 Java自带克隆模式 实现克隆模式必须实现Cloneable 接口,如果不实现会发生java.lang.CloneNotSupportedException异常 当某 ...
- 孙悟空的身外身法术使用了Java设计模式:原型模式
目录 定义 意图 主要解决问题 何时使用 优缺点 结构 简单形式的原型模式 登记形式的原型模式 两种形式比较 浅克隆和深克隆 孙悟空的身外身法术 浅克隆实现 深克隆实现 定义 原型模式属于对象的创建型 ...
- 设计模式_11_原型模式(prototype)深拷贝、浅拷贝
设计模式_11_原型模式(prototype) 浅拷贝: package designPatternOf23; /** * 定义:用原型实例,指定创建对象的种类,并通过拷贝这些原型创建新的对象 * P ...
- 【GOF23设计模式】原型模式
来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_原型模式.prototype.浅复制.深复制.Cloneable接口 浅复制 package com.test.prot ...
随机推荐
- 【python系统学习07】一张图看懂字典并学会操作
点击跳转 - 原文地址 数据类型 - 字典(dict) 目录: 一张图get字典 字典是什么 js的对象 字典长啥样 语法伪代码 示例demo 语法成像 字典怎么用 字典长度获取--len函数 提取字 ...
- 第二阶段冲刺个人任务——six
今日任务: 搭建网络服务器,上传数据库及程序. 昨日成果: 合并程序(统计团队博客).
- 18年第一弹射 和网络有关; 艾曲塞嗯诶系列篇 one
1:当指定接口运行在RIP2组播方式时,以下说法正确的是 2个答案 A 只接收RIPv2组播报文 B 不接收RIPV1 广播报文 2 下面哪条命令是把PPP的认证方式设置为PAP? C ppp au ...
- 对于Makefile的基本使用
上课不听讲的后果就是课下疯狂补知识了 原文来自https://www.cnblogs.com/chenguanfu/p/4415072.html 在Windows下,只需要简单的点击以下make,re ...
- 龙芯2f 8089D 笔记本 Debian 系统安装配置
版权声明:原创文章,未经博主允许不得转载 正文主要讲述安装社区版Debian6镜像(也有7和8,方法大同小异) 最后简单介绍了网络安装原版Debian 小记 非网络安装,没网也没事,再也不用担心网速度 ...
- rabbitmq使用总结
rabbitmq 架构图 RabbitMQ 中的 broker 是指什么?cluster 又是指什么 broker 是指一个或多个 erlang node 的逻辑分组,且 node 上运行着 Rabb ...
- REST 风格架构
------------------------------ 时间不多了, 抓紧做一些有意义的事情 REST 风格架构 1. 他是面向资源进行开发的 2. 他是基于HTTP 协议进行开发 ...
- maven 打包详解
Maven可以使用mvn package指令对项目进行打包,如果使用Java -jar xxx.jar执行运行jar文件,会出现"no main manifest attribute, in ...
- vmware安装kvm虚拟机
1. 概述 本篇博客主要使用运行在win10专业版上的vmware workstation 15 pro虚拟化软件,安装centos7.7最小化系统,并在centos7上安装kvm虚拟机,实现快速创建 ...
- 计算机原理基础:DNS
DNS服务的作用 将域名解析成IP地址 端口号:53 域名服务器 根域名服务器 所有的根域名服务器都知道所有的顶级域名服务器的域名和IP地址. 不管是哪一个本地域名服务器,若要对因特网上任何一个域名进 ...