享元模式:减少内存占用的诀窍

一,概要

享元模式(Flyweight Pattern)是一种结构型设计模式,它主要通过共享对象来降低系统中对象的数量,从而减少内存占用和提高程序性能。这听起来有点像单例模式,但它们在实现和用途上有很大的区别。享元模式的核心是把一个对象的状态分成内部状态和外部状态,内部状态即是不变的,外部状态是变化的;然后通过共享不变的部分,达到减少对象数量并节约内存的目的。享元模式的本质是缓存共享对象,降低内存消耗。

  • 内部状态:内部状态是对象可共享的部分,它存储在享元对象内部,并且不随外部环境的变化而变化。内部状态可以被多个具体享元对象共享。
  • 外部状态:外部状态是对象的上下文相关部分,它在使用时动态传入享元对象,并且随外部环境的变化而变化。外部状态不可共享,每个具体享元对象都需要接收外部状态作为参数来完成其操作。

享元模式三大角色

  • 享元接口:定义了享元对象的公共方法,这些方法可以操作享元对象的外部状态,外部状态一般作为方法参数传入。

  • 具体享元类:实现享元接口,完成具体的对象操作。内部状态作为成员属性存在,一旦初始化完成就不再改变,不对外提供setter方法。

  • 享元工厂:负责创建和管理享元对象。当客户端请求一个享元对象时,享元工厂会检查是否有已经创建的享元对象,如果有,则直接返回;如果没有,则创建一个新的享元对象并加入到享元池中。

优点

  1. 减少内存占用:相同对象只要保存一份,大大降低了系统中对象的数量。
  2. 提高性能:享元模式减少了对象的创建和销毁,降低了垃圾回收的开销,从而提高了程序性能。
  3. 提高可扩展性:通过外部状态的引入,享元模式可以灵活地处理不同的上下文,使得系统更具可扩展性。

缺点

  1. 复杂度增加:享元模式需要将对象的状态进行拆分,引入了内部状态和外部状态的管理,增加了系统的复杂性。

  2. 线程安全问题:由于享元对象是共享的,因此在多线程环境下,对享元对象的操作需要考虑线程安全。

适用场景

  • 系统中存在大量细粒度的对象,且这些对象的状态可以分为内部状态和外部状态时,可以考虑使用享元模式。
  • 对象的大部分状态可以共享,而一小部分状态需要外部环境来改变时,可以使用享元模式。
  • 需要缓存对象以提高系统性能,并且可以接受一定的对象复用时,可以使用享元模式。
  • 需要对对象进行池化管理,以便于统一控制和管理对象的创建、销毁和状态维护时,可以考虑使用享元模式。

二,实现案例

假设我们需要在一个坐标图纸上,绘制100个固定半径的圆,圆分为三种颜色。常规方法,我们定义一个圆,里面包含半径,颜色,横坐标,纵坐标四个属性,再定义一个绘制的方法draw(),然后我们创建100个对象,调用draw()方法就可以实现。但是如果我们用享元模式实现,仅需创建3个对象即可,其关键就是将颜色和半径作为内部状态共享,将坐标作为外部状态分离出来。

步骤1:创建享元接口Shape

public interface Shape {//x,y表示坐标void draw(int x,int y);
}

步骤2:创建具体享元类Circle

public class Circle implements Shape{private String color;private int radius;public Circle(String color){this.color = color;this.radius = 10;}@Overridepublic void draw(int x,int y) {System.out.println("--在坐标("+x+","+y+")处画圆: [颜色 : " + color+", 半径 :" + radius+"]");}
}

步骤3:创建享元工厂ShapeFactory

public class ShapeFactory {private static final HashMap<String, Shape> circleMap = new HashMap<>();public static Shape getCircle(String color) {Circle circle = (Circle)circleMap.get(color);System.out.println("从缓存中获取"+color+"色的圆");if(circle == null) {circle = new Circle(color);circleMap.put(color, circle);System.out.println("缓存中不存在,先创建"+color+"色的圆,并放入缓存中");}return circle;}
}

步骤4:客户端测试

public class Client {private static final String colors[] = { "Red", "Green", "Blue" };public static void main(String[] args) {for(int i=0; i < 10; ++i) {Circle circle = (Circle)ShapeFactory.getCircle(getRandomColor());circle.draw(getRandomX(),getRandomY());}}private static String getRandomColor() {return colors[(int)(Math.random()*colors.length)];}private static int getRandomX() {return (int)(Math.random()*100 );}private static int getRandomY() {return (int)(Math.random()*100);}
}

测试结果

image-20230608112221138

三,总结

享元模式是一种非常实用的设计模式,它的思想很简单,就是把一些可以共享的对象只创建一份,放入到缓存池中,供业务方引用。这样做可以大大减少对象的创建开销,减少内存中相似或相同对象的数量,减少内存占用。在java源码中也有很多享元模式的思想体现,如String的字符串常量池,Integer包装类中的IntegerCach,以及各种连接池,线程池等技术。我们在日常开发过程中,可以结合上下文场景,灵活运用享元模式,但需要注意线程安全性和共享池的管理。

最后,希望这篇文章能对大家有所帮助,如果你喜欢这篇文章,不妨点个赞和分享给你的朋友们。欢迎大家在评论区留言交流,我们下次再见!👋
在这里插入图片描述

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

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

相关文章

web前端求职面试题参考精选(合集)

web前端求职面试题之选择题 1. CSS样式表根据所在网页的位置&#xff0c;可分为?(B ) A.行内样式表、内嵌样式表、混合样式表 B.行内样式表、内嵌样式表、外部样式表 C.外部样式表、内嵌样式表、导入样式表 D.外部样式表、混合样式表、导入样式表 2. 对于标签&#xff0…

01 | 一条 SQL 查询语句是如何执行的?

以下内容出自 《MySQL 实战 45 讲》 一条 SQL 查询语句是如何执行的&#xff1f; 下面是 MySQL 的基本架构示意图&#xff0c;从中可以清楚地看到 SQL 语句在 MySQL 的各个功能模块中的执行过程。 大体来说&#xff0c;MySQL 可以分为 Server 层和存储引擎层两部分。 Server …

较旧系统的轻量级的LINUX发行版—FATDOG64

导读我们回顾FatDog64 Linux&#xff0c;这个轻量级的Linux发行版可能不是现代的&#xff0c;但它非常适合于老化的计算机&#xff0c;fatdog64 Linux是一个小而灵活的64位多用户Linux发行版。 有一个Linux发行版几乎每一个都需要&#xff0c;有Linux分布的黑客&#xff0c;天…

Jenkins在Ubuntu的安装问题

使用apt安装没有成功&#xff0c;各种报错。最后使用了离线安装方式。 1、安装jdk。和之前的安装jdk无异&#xff0c;增加一步 添加一个软链接 sudo ln -s /path/to/java/home/bin/java /usr/bin/java 2、下载deb包&#xff0c;然后安装 2.1、前置步骤&#xff0c;安装可能…

点击向数组中添加对象,并判断是否已经包含了重复值

需求&#xff1a;现在有一堆列表&#xff0c;当我每次点击列表的每一项时&#xff0c;希望将此项添加到一个数组中&#xff0c;并且已添加的数据不能再添加 效果图 实现&#xff1a; <!DOCTYPE html> <html lang"zh"> <head><meta charset&quo…

kafka3.x 入门 安装(一)

一、下载地址 http://kafka.apache.org/downloads.html 二、 zookeeper安装教程 https://blog.csdn.net/weixin_43205308/article/details/130426019 三、解压 这里使用的是kafka_2.12-3.0.0.tgz tar -zvxf kafka_2.12-3.0.0.tgz四、修改配置 进入到安装路径的config文件夹下…

chatgpt赋能python:Python选择排序最简单的写法是什么?

Python选择排序最简单的写法是什么&#xff1f; 选择排序是一种简单的排序算法&#xff0c;通常用于小规模数据集的排序。Python是一种广泛使用的脚本语言&#xff0c;也可以用来实现选择排序算法。这篇文章将介绍Python选择排序的最简单写法&#xff0c;并分析其优缺点。 选…

Android设计模式—桥接模式

1.桥接模式 桥接模式是一种结构型设计模式&#xff0c;它通过将抽象部分与实现部分分离来解耦。它使用接口作为桥梁&#xff0c;将一个抽象类与其实现类的代码独立开来&#xff0c;从而使它们可以各自独立地变化。桥接模式的核心思想是“组合优于继承”。 简单来讲&#xff0…

dbca添加实例时无法发现对方主机

有个12.2集群环境&#xff0c;需要添加一个实例&#xff0c;直接图形化启动dbca&#xff0c;第四步时只能发现本机&#xff0c;无法识别到另外一个节点&#xff0c;如下图 通过排查发现是oracle用户的密码过期了&#xff0c;细节如下 [roothydb1 ~]# su - grid Last login: W…

ASCII码完整版对照表,收藏起来以备不时之需

目录 ASCII控制字符对照表 ASCII可显示字符对照表 ASCII扩展字符对照表 ASCII编码即美国信息交换标准代码&#xff08;American Standard Code for Information Interchange&#xff09;是一套共有128个字符的编码&#xff0c;它基于阿拉丁字母&#xff0c;主要作用是用来表…

idea生成serialVersionUID序列号

设置idea file->settings,搜索serialVersionUID&#xff0c;勾选框起来的两项 实体类实现Serializable接口 Data public class User implements Serializable { }鼠标放到类名上 点击提示的uid 生成的uid 结束&#xff01; hy:17 生活是一面镜子&#xff0c;给予我们…

4-移动端适配-2

01-vw适配方案 vw和vh基本使用 vw和vh是相对单位&#xff0c;相对视口尺寸计算结果 vw&#xff1a;viewport width&#xff08;1vw 1/100视口宽度 &#xff09;vh&#xff1a;lviewport height ( 1vh 1/100视口高度 ) vw布局 vw单位的尺寸 px 单位数值 / ( 1/100 视口宽…