原型模式与享元模式:提升系统性能的利器

        原型模式和享元模式,前者是在创建多个实例时,对创建过程的性能进行调优后者是用减
少创建实例的方式,来调优系统性能
。这么看,你会不会觉得两个模式有点相互矛盾呢?

          

        在有些场景下,我们需要重复创建多个实例,例如在循环体中赋值一个对象此时我们就可以采用原型模式来优化对象的创建过程;而在有些场景下,我们则可以避免重复创建多个实例,在内存中共享对象就好了    

 原型模型

原型模型的实现

原型模式是通过给出一个原型对象来指明所创建的对象的类型,然后使用自身实现的克隆接口来复制这个原型对象,该模式就是用这种方式来创建出更多同类型的对象。

使用这种方式创建新的对象的话,就无需再通过 new 实例化来创建对象了。这是因为Object 类的 clone 方法是一个本地方法,它可以直接操作内存中的二进制流,所以性能相对 new 实例化来说,更佳

new一个对象和clone一个对象,性能差在哪里呢?上边提到直接从内存复制二进制怎末理解?


答: 一个对象通过new创建的过程为:
1、在内存中开辟一块空间;
2、在开辟的内存空间中创建对象;
3、调用对象的构造函数进行初始化对象。
而一个对象通过clone创建的过程为:
1、根据原对象内存大小开辟一块内存空间;
2、复制已有对象,克隆对象中所有属性值。
相对new来说,clone少了调用构造函数。如果构造函数中存在大量属性初始化或大对象,则使用clone的复制对象的方式性能会好一些。

首先先看一段原型模式的代码

要实现一个原型类,需要具备三个条件:

实现 Cloneable 接口:Cloneable 接口与序列化接口的作用类似,它只是告诉虚拟机可
以安全地在实现了这个接口的类上使用 clone 方法。在 JVM 中,只有实现了 Cloneable
接口的类才可以被拷贝,否则会抛出 CloneNotSupportedException 异常。
重写 Object 类中的 clone 方法:在 Java 中,所有类的父类都是 Object 类,而 Object
类中有一个 clone 方法,作用是返回对象的一个拷贝。
在重写的 clone 方法中调用 super.clone():默认情况下,类不具备复制对象的能力,需
要调用 super.clone() 来实现

从上面我们可以看出,原型模式的主要特征就是使用 clone 方法复制一个对象。通常,有
些人会误以为 Object a=new Object();Object b=a; 这种形式就是一种对象复制的过程,
然而这种复制只是对象引用的复制,也就是 a 和 b 对象指向了同一个内存地址,如果 b 修
改了,a 的值也就跟着被修改了。我们可以通过一个简单的例子来看看普通的对象复制问题:

 如果是复制对象,此时打印的日志应该为:

 然而,实际上是:

学生 1: test2

学生 2 :test2

通过 clone 方法复制的对象才是真正的对象复制,clone 方法赋值的对象完全是一个独立
的对象
。Object 类的 clone 方法是一个本地方法,它直接操作内存中的二进
制流,特别是复制大对象时,性能的差别非常明显。我们可以用 clone 方法再实现一遍以
上例子

// 学生类实现 Cloneable 接口
class Student implements Cloneable{
private String name; // 姓名
public String getName() {
return name;
}
public void setName(String name) {
this.name= name;
}
// 重写 clone 方法
public Student clone() {
Student student = null;
try {
student = (Student) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return student;
}
}
public class Test {
public static void main(String args[]) {
Student stu1 = new Student(); // 创建学生 1
stu1.setName("test1");
Student stu2 = stu1.clone(); // 通过克隆创建学生 2
stu2.setName("test2");
System.out.println(" 学生 1:" + stu1.getName());
System.out.println(" 学生 2:" + stu2.getName());
}
}

 运行结果:

学生 1:test1

学生 2:test2

深拷贝和浅拷贝

        在调用 super.clone() 方法之后,首先会检查当前对象所属的类是否支持 clone,也就是看
该类是否实现了 Cloneable 接口

        如果支持,则创建当前对象所属类的一个新对象,并对该对象进行初始化,使得新对象的成
员变量的值与当前对象的成员变量的值一模一样,但对于其它对象的引用以及 List 等类型
的成员属性,则只能复制这些对象的引用了。所以简单调用 super.clone() 这种克隆对象方
式,就是一种浅拷贝。

        所以,当我们在使用 clone() 方法实现对象的克隆时,就需要注意浅拷贝带来的问题。我们
再通过一个例子来看看浅拷贝。

// 定义学生类
class Student implements Cloneable{
private String name; // 学生姓名
private Teacher teacher; // 定义老师类
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Teacher getTeacher() {
return teacher;
}
public void setName(Teacher teacher) {
this.teacher = teacher;
}
//重写克隆方法
public Student clone() {
Student student = null;
try {
student = (Student) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return student;
}
}
// 定义老师类
class Teacher implements Cloneable{
private String name; // 老师姓名
public String getName() {
return name;
}
public void setName(String name) {
this.name= name;
}
// 重写克隆方法,堆老师类进行克隆
public Teacher clone() {
Teacher teacher= null;
try {
teacher= (Teacher) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return student;
}
}
public class Test {
public static void main(String args[]) {
Teacher teacher = new Teacher (); // 定义老师 1
teacher.setName(" 刘老师 ");
Student stu1 = new Student(); // 定义学生 1
stu1.setName("test1");
stu1.setTeacher(teacher);
Student stu2 = stu1.clone(); // 定义学生 2
stu2.setName("test2");
stu2.getTeacher().setName(" 王老师 ");// 修改老师
System.out.println(" 学生 " + stu1.getName + " 的老师是:" + stu1.getTeacher().get
System.out.println(" 学生 " + stu1.getName + " 的老师是:" + stu2.getTeacher().get
}
}

运行结果:

观察以上运行结果,我们可以发现:在我们给学生 2 修改老师的时候,学生 1 的老师也跟
着被修改了。这就是浅拷贝带来的问题。 

我们可以通过深拷贝来解决这种问题,其实深拷贝就是基于浅拷贝来递归实现具体的每个对
,代码如下:

 适用场景

在一些重复创建对象的场景下,我们就可以使用原型模式来提高对象的创建性能。例如,在开头提到的,循环体内创建对象时,我们就可以考虑用 clone 的方式来实现。

例如:

可以优化为 :

 除此之外,原型模式在开源框架中的应用也非常广泛。例如 Spring 中,@Service 默认都
是单例的。用了私有全局变量,若不想影响下次注入或每次上下文获取 bean,就需要用到
原型模式,我们可以通过以下注解来实现,@Scope(“prototype”)。

以上可能会有这个哥们们不理解比如这个问题

文中说的,@service默认是单例模式,若不想影响下次请求,就要使用原型模式?

回复:纠正下,不是每次请求,而是每次bean注入或通过上下文获取bean时。

如果我们使用的是单例,假设有一个全局变量private int a=1,我们通过上下文获取到实例,调用A方法修改了变量a=2,此时下一个通过上下文获取到实例调用B方法获取变量,则a=2。

如果我们使用的是原型模式,假设有一个全局变量private int a=1,我们通过上下文获取到实
例,调用A方法修改了变量a=2,此时下一个通过上下文获取到实例调用B方法获取变量,则还是a=1。

享元模式

        享元模式是运用共享技术有效地最大限度地复用细粒度对象的一种模式。该模式中,以对象
的信息状态划分,可以分为内部数据和外部数据内部数据是对象可以共享出来的信息,这
些信息不会随着系统的运行而改变;外部数据则是在不同运行时被标记了不同的值。

        享元模式一般可以分为三个角色,分别为 Flyweight(抽象享元类)、
ConcreteFlyweight(具体享元类)和 FlyweightFactory(享元工厂类)。抽象享元类通
常是一个接口或抽象类
,向外界提供享元对象的内部数据或外部数据具体享元类是指具体
实现内部数据共享的类;享元工厂类则是主要用于创建和管理享元对象的工厂类


 

 

 

 

 这里注意内在数据,也就是线程共享的,这里咱们定义的是私有的String,大家会有欸他不是静态的怎末共享啊。???????????????????????

 

观察以上代码运行结果,我们可以发现:如果对象已经存在于享元池中,则不会再创建该对
象了,而是共用享元池中内部数据一致的对象。这样就减少了对象的创建,同时也节省了同
样内部数据的对象所占用的内存空间。

适用场景

享元模式在实际开发中的应用也非常广泛。例如 Java 的 String 字符串,在一些字符串常
量中,会共享常量池中字符串对象,从而减少重复创建相同值对象,占用内存空间。代码如
下:

 

 在日常开发中的应用。例如,线程池就是享元模式的一种实现;将商品存储在应用服
务的缓存中,那么每当用户获取商品信息时,则不需要每次都从 redis 缓存或者数据库中获
取商品信息,并在内存中重复创建商品信息了。

  • 总结

在不得已需要重复创建大量同一对象时,我们可以使用原型模式,通过 clone 方法复制对
象,这种方式比用 new 和序列化创建对象的效率要高;在创建对象时,如果我们可以共用
对象的内部数据,那么通过享元模式共享相同的内部数据的对象,就可以减少对象的创建,
实现系统调优

一些杂谈

  • 线上短信业务被轰炸,流量费倍增……求推荐个解决思路,监测发现是爬虫程序

        建议加一个图片验证码

  • 这里需要考虑一个问题就是享元模式,在工厂哪用的那个hashmap的方式

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/30357.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

23款奔驰GLS450时尚型升级原厂香氛负离子系统,清香宜人,久闻不腻

奔驰原厂香氛合理性可通过车内空气调节组件营造芳香四溢的怡人氛围。通过更换手套箱内香氛喷雾发生器所用的香水瓶,可轻松选择其他香氛。香氛的浓度和持续时间可调。淡雅的香氛缓缓喷出,并且在关闭后能够立刻散去。车内气味不会永久改变,香氛…

解数独(Java)

题目链接: 力扣 题目详情: 37. 解数独t编写一个程序,通过填充空格来解决数独问题。 数独的解法需 遵循如下规则: 数字 1-9 在每一行只能出现一次。数字 1-9 在每一列只能出现一次。数字 1-9 在每一个以粗实线分隔的 3x3 宫内只…

nodejs实现解析chm文件列表,无需转换为PDF文件格式,在线预览chm文件以及目录,不依赖任何网页端插件

特性: 1、支持任意深度的chm文件解析 2、解析后内容结构转换为tree数据呈现 3、点击树节点可以在html实时查看数据 4、不依赖任何浏览器端插件,兼容性较好 nodejs端核心代码 const $g = global.SG.$g, fs = global.SG.fs, router = global.SG.router, xlsx = global.SG.xl…

【ADS】在同一个symbol层次化实现理想器件和EM仿真的切换

ADS层次化切换理想器件和电磁仿真 1.需求描述2.实现步骤2.1 层次结构2.2 新建schematic2.3 新建symbol2.4 使用演示 3.后言 1.需求描述 在使用ADS做电磁仿真时,得到的结果需要多次迭代去接近原理图,那么就需要反复与原理图切换进行对比。 比较直接的方法…

canvas实现代码雨

学习抖音&#xff1a; 渡一前端必修课 效果图&#xff1a; 全部代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge">&…

flinksql sink to sr often fail because of nullpoint

flinksql or DS sink to starrocks often fail because of nullpoint flink sql 和 flink ds sink starrocks 经常报NullpointException重新编译代码 并上传到flink 集群 验证&#xff0c;有效 flink sql 和 flink ds sink starrocks 经常报NullpointException 使用flink-sta…

PHP流浪动物招领网站mysql数据库web结构apache计算机软件工程网页wamp

一、源码特点 PHP流浪动物招领网站 是一套完善的web设计系统&#xff0c;对理解php编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。 下载链接 nullhttps://download.csdn.net/download/qq_41221322/88190168视频演示 …

观察混合云环境中 Kubernetes 可观测性的 6 种有效策略

在混合云环境中观察Kubernetes需要理解分布式系统的行为和性能。我下面这篇文章中的六个策略可以帮助实现这一目标。 2023年&#xff0c;原生云应用和平台迅速增长。组织不断努力最大化其应用程序的潜力&#xff0c;确保无缝的用户体验&#xff0c;并推动业务增长。混合云环境…

O2OA开发平台实施入门指南

O2OA&#xff08;翱途&#xff09;开发平台&#xff0c;是一款适用于协同办公系统开发与实施的基础平台&#xff0c;说到底&#xff0c;它也是一款快速开发平台。开发者可以基于平台提供的能力完成门户、流程、信息相关的业务功能开发。 既然定位为开发平台&#xff0c;那么开…

RocketMQ 5.x如何使用GRPC方式发送消费消息

这里是weihubeats,觉得文章不错可以关注公众号小奏技术&#xff0c;文章首发。拒绝营销号&#xff0c;拒绝标题党 RocketMQ版本 5.1.0 背景 我们都知道RocketMQ 5.x新增了proxy模式部署方式&#xff0c;也就是支持了GRPC的消费方式消费&#xff0c;所以今天我们来试试 本次…

SAN共享存储架构

SAN共享存储架构 概述 近年在高性能专用存储网络需求的驱使下&#xff0c;许多SAN存储系统应用于高性能计算网络系统、大型网站系统、非线性编辑系统等网络系统中&#xff0c;存储设备与计算机主机系统之间一对一的关系&#xff0c;被可供多个计算机主机共享读写同一个存储设…

探究Vue源码:mustache模板引擎(10) 解决不能用连续点符号找到多层对象问题,为编译循环结构做铺垫

上文 探究Vue源码:mustache模板引擎(9) 将单层无喜欢结果tokens转为dom字符串 我们简单处理了 token转字符串的业务逻辑 但是 我们只处理了最贱的花括号 接下来 带着大家将井号的也处理一下 我们打开项目 将 www中的index.html代码改回之前的这样 <!DOCTYPE html> <h…

改进的麻雀算法优化最大相关峭度解卷积(SCSSA-MCKD),实现早期微弱故障诊断,MATLAB代码实现

01 引言 由于一些设备的早期故障产生的冲击十分微弱&#xff0c;易被系统噪声干扰&#xff0c;如何有效地对设备的原始故障信号进行降噪并增强信号中微弱冲击成分&#xff0c;是进行该类部件早期故障诊断的关键。 最大相关峭度解卷积&#xff08;MCKD&#xff09;通过解卷积运算…

leetcode 399-除法求值

法一&#xff1a;并查集 分析示例1&#xff1a; a / b 2.0 a/ b 2.0 a/b2.0&#xff0c;说明 a 2 b a2b a2b&#xff0c; a a a和 b b b在同一个集合中 b / c 3.0 b/c3.0 b/c3.0&#xff0c;说明 b 3 c b3c b3c&#xff0c; b b b和 c c c在同一个集合中 求 a / c a/…

24届近5年重庆邮电大学自动化考研院校分析

今天给大家带来的是重庆邮电大学控制考研分析 满满干货&#xff5e;还不快快点赞收藏 一、重庆邮电大学 学校简介 重庆邮电大学简称"重邮"&#xff0c;坐落于直辖市-重庆市&#xff0c;入选国家"中西部高校基础能力建设工程”、国家“卓越工程师教育培养计划…

c51单片机16个按键密码锁源代码(富proteus电路图)

注意了&#xff1a;这个代码你是没法直接运行的&#xff0c;但是如果你看得懂&#xff0c;随便改一改不超过1分钟就可以用 #include "reg51.h" #include "myheader.h" void displayNumber(unsigned char num) {if(num1){P10XFF;P10P11P14P15P160;}else if…

PyCharm新手入门指南

安装好Pycharm后&#xff0c;就可以开始编写第一个函数&#xff1a;Hello World啦~我们就先来学习一些基本的操作&#xff0c;主要包含新建Python文件&#xff0c;运行代码&#xff0c;查看结果等等。 文章主要包含五个部分&#xff1a; 一、界面介绍 主要分为菜单栏、项目目录…

开发工具Eclipse的使用之导入项目(import)

&#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 接下来看看由辉辉所写的关于Eclipse使用的相关操作吧 目录 &#x1f973;&#x1f973;Welcome Huihuis Code World ! !&#x1f973;&#x1f973; 一.导读 二.详细操作步骤 1.右击项…

吐血整理,Python接口自动化测试-接口关联依赖处理(详细)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 场景说明 在面试…

cpolar的基础使用方法

如何使用cpolar内网穿透&#xff1f; 文章目录 如何使用cpolar内网穿透&#xff1f;前言1. 在群辉NAS系统下安装cpolar套件2. 管理隧道列表3. 创建固定数据隧道 前言 群晖作为大容量存储系统&#xff0c;既可以作为个人的私有存储设备&#xff0c;也可以放在小型企业中作为数据…