当前位置: 首页 > news >正文

设计模式-- 原型模式详解

原型模式(prototype)

原型模式:用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型相同或相似的新对象,原型模式属于创造性模式,它同样提供了创建对象的最佳方式之一。(效率很高

原型模式实现了一个原型接口,该接口用于创建当前对象的克隆,当创建的对象过于复杂,代价较大的时候,使用原型模式,

例如,一个对象需要在一个高代价的数据库操作之后被创建。我们可以缓存该对象,在下一个请求时返回它的克隆,在需要的时候更新数据库,以此来减少数据库调用。

原型模式的实现

Java提供了对象的clone()方法,实现原型模式只需要实现Cloneable接口,重写clone()方法

原型模式包括:

  • 抽象原型类:规定了具体原型对象必须实现的接口。

  • 具体原型类:实现抽象原型类的 clone() 方法,它是可被复制的对象。

  • 客户端:使用具体原型类中的 clone() 方法来复制新的对象。

代码展示:

具体原型类:

 package com.lyc.prototype.demo1;​import lombok.*;​import java.util.Date;/** 1.实现一个接口 Cloneable* 2,重写一个方法 clone()* *///视频的模型@Getter@Setter@ToString@AllArgsConstructor@NoArgsConstructorpublic class Video implements Cloneable{//抄袭者,抄别人的视频private String name;private Date createTime;​@Overrideprotected Object clone() throws CloneNotSupportedException {return super.clone();}}

客户端:

 package com.lyc.prototype.demo1;​import java.util.Date;​//客户端public class Client {public static void main(String[] args) throws Exception {//原型对象 video//浅克隆 video和video2的createTime是同一个对象,date修改,video2也会变Date date = new Date();Video video = new Video("设计模式之原型模式",date);Video video2 = (Video) video.clone();System.out.println("video:"+video);System.out.println("video2:"+video2);date.setTime(7756796);System.out.println("video:"+video);System.out.println("video2:"+video2);​//        video:Video(name=设计模式之原型模式, createTime=Fri Apr 25 16:55:46 CST 2025)//        hashCode:284720968//        video2:Video(name=设计模式之原型模式, createTime=Fri Apr 25 16:55:46 CST 2025)//        hashCode:2093176254}/**   System.out.println("video:"+video);System.out.println("hashCode:"+video.hashCode());//video 克隆 video2 克隆出来的对象和原来是一模一样的System.out.println("video2:"+video2);System.out.println("hashCode:"+video2.hashCode());* */}
拓展知识

native

Java关键字,是用来说明该方法是原生函数,即这个方法使用C/C++语言实现的,并且被编译成了DLL,由Java调用。

native的意思就是通知操作系统,这个函数必须实现,所以native关键字的函数都是操作系统实现的,Java只能调用。

java是跨平台的语言,既然是跨了平台,所付出的代价就是牺牲一些对底层的控制,而java要实现对底层的控制,就需要一些其他语言的帮助,这个就是native的作用了

Java不是完美的,Java的不足除了体现在运行速度上要比传统的C++慢许多之外,Java无法直接访问到操作系统底层(如系统硬件等),为此Java使用native方法来扩展Java程序的功能。
  可以将native方法比作Java程序同C程序的接口,其实现步骤:

  1. 在Java中声明native()方法,然后编译;
  2. 用javah产生一个.h文件;
  3. 写一个.cpp文件实现native导出方法,其中需要包含第二步产生的.h文件(注意其中又包含了JDK带的jni.h文件);
  4. 将第三步的.cpp文件编译成动态链接库文件;
  5. 在Java中用System.loadLibrary()方法加载第四步产生的动态链接库文件,这个native()方法就可以在Java中被访问了。

原型模式优点

  • 性能优良:原型模式是使用native本地方法clone(),直接从内存中二进制流的拷贝,要比new一个对象性能好很多,特别是在一个循环体类产生大量对象的时候更加明显。

  • 逃避构造函数的约束:这是优缺点共存的一点,直接在内存中拷贝,构造函数是不会执行的

缺点:

  • 配备克隆方法需要全面考虑类的功能,对已有类可能较难实现,特别是处理不支持串行化的间接对象或含有循环结构的引用时。

  • 必须实现 Cloneable 接口

应用实例

JavaScript对象的继承就是使用原型链来完成的。

Spring中Bean的创建分为单例模式,原型模式。

适用场景

  • 资源优化(直接调用本地方法,节省资源,效率较高)

  • 类初始化需要消耗大量资源(如数据、硬件资源)

  • 性能和安全要求高的场景(直接从底层调用方法,性能很高,clone方法直接拷贝对象,安全性高)

  • 通过 new 创建对象需要复杂的数据准备或访问权限时

  • 一个对象需要多个修改者

  • 对象需提供给其他对象访问并可能被各个调用者修改时

  • 通常与工厂方法模式一起使用,通过 clone 创建对象,然后由工厂方法提供给调用者

拓展:浅拷贝 与 深拷贝

浅拷贝:在clone之后,两个对象共用一个私有属性,该属性性改变则两个对象一块改变

原因:object类的clone方法只拷贝本对象,其对象内部的数组,引用对象等都不拷贝,还是指向原生对象的内部元素地址,这种拷贝就是浅拷贝。两个对象共用一个私有变量。这是一种非常不安全的方式。 

原始类型会被拷贝(int,double,long。。。),String类型也会被拷贝;数组、引用类型不会被拷贝

代码展示:

 public static void main(String[] args) throws Exception {//原型对象 video//浅克隆 video和video2的createTime是同一个对象,date修改,video2也会变Date date = new Date();Video video = new Video("设计模式之原型模式",date);Video video2 = (Video) video.clone();System.out.println("video:"+video);System.out.println("video2:"+video2);date.setTime(7756796);System.out.println("==============================================");System.out.println("video:"+video);System.out.println("video2:"+video2);

效果展示:

 video:Video(name=设计模式之原型模式, createTime=Fri Apr 25 17:50:08 CST 2025)video2:Video(name=设计模式之原型模式, createTime=Fri Apr 25 17:50:08 CST 2025)==============================================video:Video(name=设计模式之原型模式, createTime=Thu Jan 01 10:09:16 CST 1970)video2:Video(name=设计模式之原型模式, createTime=Thu Jan 01 10:09:16 CST 1970)

深拷贝:在clone之后,两个对象的属性也被克隆,不再共用

如何实现?

  • 方式一:重写clone方法,将对象的属性也克隆一份

  • 方式二:通过实现 Serializable 读取二进制流实现。

方式一实现方式:

 package com.lyc.prototype.demo2;​import lombok.*;​import java.util.Date;/** 1.实现一个接口 Cloneable* 2,重写一个方法 clone()* *///视频的模型@Getter@Setter@ToString@AllArgsConstructor@NoArgsConstructorpublic class Video implements Cloneable{//抄袭者,抄别人的视频private String name;private Date createTime;​@Overrideprotected Object clone() throws CloneNotSupportedException {//实现深克隆的方式一,方式二: 序列化,反序列化Object clone = super.clone();Video v = (Video)clone;//将该对象的属性也克隆v.createTime = (Date) createTime.clone();return clone;}}
 public class Client {public static void main(String[] args) throws Exception {//原型对象 video//浅克隆 video和video2的createTime是同一个对象,date修改,video2也会变Date date = new Date();Video video = new Video("设计模式之原型模式",date);Video video2 = (Video) video.clone();System.out.println("video:"+video);System.out.println("video2:"+video2);date.setTime(7756796);System.out.println("==============================================");System.out.println("video:"+video);System.out.println("video2:"+video2);

效果展示:

 video:Video(name=设计模式之原型模式, createTime=Fri Apr 25 17:54:20 CST 2025)video2:Video(name=设计模式之原型模式, createTime=Fri Apr 25 17:54:20 CST 2025)==============================================video:Video(name=设计模式之原型模式, createTime=Thu Jan 01 10:09:16 CST 1970)video2:Video(name=设计模式之原型模式, createTime=Fri Apr 25 17:54:20 CST 2025)

注意事项:

  • 构造方法在clone的时候并不会执行,因为对象是从内存以二进制流的方式进行拷贝,当然不会执行

  • 深拷贝和浅拷贝要分开实现,不然会导致程序变得非常复杂

  • 带有final类型的变量是不可以进行拷贝的,这样是无法实现深拷贝

  这是因为final关键字的特性

  对于一个final变量,如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象。因此:要使用clone()方法,类的成员变量上不要增加final关键字。

小结:

  使用原型类型时,引用的成员变量必须满足两个条件才不会被拷贝:

  • 是类的成员变量,而不是方法内的变量

  • 必须是一个可变的引用对象,而不是一个原始类型或者不可变对象(比如final)

以上就是我对原型模式的理解,希望对大家有所帮助

相关文章:

  • mybatis-plus里的com.baomidou.mybatisplus.core.override.MybatisMapperProxy 类的详细解析
  • 【Linux网络】:套接字之UDP
  • 《免费开放”双刃剑:字节跳动Coze如何撬动AI生态霸权与暗涌危机?》
  • 交叉编译paho.mqtt.c和paho.mqtt.cpp(MQTT客户端)
  • 制作一款打飞机游戏25:添加数据
  • 芯岭技术XL32F003单片机 32位Cortex M0+ MCU简单介绍 性能优异
  • 【Harmony_Bug】forEach + asyncawait 的异步陷阱
  • node.js 实战——从0开始做一个餐厅预订(express+node+ejs+bootstrap)
  • git 的基本使用
  • RTMP 入门指南
  • 【语法】C++的stack和queue
  • 零基础快速搭建AI绘画网站!用Gradio玩转Stable Diffusion
  • Langchain入门介绍
  • Spring @Transactional 自调用问题深度解析
  • C语言----函数栈帧讲解
  • 抖音IP属地跟无线网有关系吗?一文解析
  • 【智慧城市】新中地GIS开发实训:基于 Vue 框架的武汉智慧旅游系统
  • http协议详解附带c/c++服务器和客户端源码
  • 在统信UOS/麒麟Kylin OS中创建网页桌面快捷方式
  • 反序列化漏洞1
  • 屋顶上的阳光与火光:战争如何改变了加沙的能源格局
  • 六部门:进一步优化离境退税政策扩大入境消费
  • 特朗普将举行集会庆祝重返白宫执政百日,美媒:时机不当
  • 新希望去年归母净利润4.74亿同比增逾九成,营收降27%
  • 铜钴巨头洛阳钼业一季度净利润同比大增九成,最新宣布超30亿元收购黄金资产
  • 光线传媒:正与部分重点地区洽谈主题乐园合作,首款3A游戏预计三年左右推出