秒懂设计模式--学习笔记(11)【结构型-享元模式】
目录
- 10、享元模式
- 10.1 享元模式
- 10.2 举例
- 10.2.1 马赛克
- 10.2.2 游戏地图(以草原地图作为范例)
- 10.3 总结
10、享元模式
10.1 享元模式
- “享元”则是共享元件的意思
- 享元模式的英文flyweight是轻量级的意思,这就意味着享元模式能使程序变得更加轻量化
- 当系统存在大量的对象,并且这些对象又具有相同的内部状态时,我们就可以用享元模式共享相同的元件对象,以避免对象泛滥造成资源浪费。
- 测试类结构
10.2 举例
10.2.1 马赛克
- 虽然马赛克小块数量比较多,但经过观察我们会发现,
- 分析组成,进行归类后,找到元件只有4种:黑色块、灰色块、灰白色块以及白色块。
- 我们可以说,这就是4个“元”色块
10.2.2 游戏地图(以草原地图作为范例)
- 对组成部分进行归类分析,找到原件
- 游戏地图都是由一个个小的单元图块组成的
- 其中除房屋比较大之外,其他图块的尺寸都一样,它们分别为河流、草地、道路,这些图块便是4个元图块
- 分析建模
- 定义一个图块类来描述图块,具体属性应该包括“图片”和“位置”信息,并且具备按照这些信息去绘制图块的能力:Segment
package flyweight;/*** @Description 图块类*/ public class Segment {/*** 材质图*/private String image;/*** 位置坐标*/private int x,y;/*** 显式带参构造方法:初始化各参数* @param image 材质图* @param x 横坐标* @param y 纵坐标*/public Segment(String image, int x, int y) {this.image = image;System.out.println("从磁盘加载[" + image + "]图片……");this.x = x;this.y = y;}/*** 图块绘制方法:按坐标位置绘制在地图上*/public void draw() {System.out.println("在位置[" + x + ":" + y + "]上绘制图片:[" + image + "]");} }
- 在地图第一行随便绘制一些图块,Client.test1()
- 在这一步会发现,图片加载很慢,一张图片加载要半秒,10张图块就要耗费5秒,影响用户体验
package flyweight;/** *@Description 测试类 */ public class Client {private static void test1() {//在地图第一行随便绘制一些图块new Segment("河流", 10, 10).draw();new Segment("河流", 10, 20).draw();new Segment("道路", 10, 30).draw();new Segment("草地", 10, 40).draw();new Segment("草地", 10, 50).draw();new Segment("草地", 10, 60).draw();new Segment("草地", 10, 70).draw();new Segment("草地", 10, 80).draw();new Segment("道路", 10, 90).draw();new Segment("道路", 10, 100).draw();} }
- 图片与坐标状态初始化后就固定下来了,简单讲就是被绘制出来后就不必变动了,即使要变也是将拼好的地图作为一个大对象整体挪动
- 图件共享(优化)
- 继续分析每个图块的坐标是不同的,但有很大一部分图块的材质图(图片)是相同的
- 于是我们可以得出结论,材质图是可以作为享元的,而坐标则不能
- 既然要共享相同的图片,那么我们就得将图块类按图片拆分成更细的材质类,如河流类、草地类、道路类等
- 而坐标不能作为图块类的享元属性,所以我们就得设法把这个属性抽离出去由外部负责
- 代码实战
- 首先需要定义一个接口,规范这些材质类的绘图标准(接口:规范标准Drawable)
- 当然,除了接口方式,我们还可以用抽象类抽离出更多的属性和方法,使子类变得更加简单
- 首先需要定义一个接口,规范这些材质类的绘图标准(接口:规范标准Drawable)
package flyweight; /** * @Description 绘图接口: 规范这些材质类的绘图标准 * 当然,除了接口方式,我们还可以用抽象类抽离出更多的属性和方法,使子类变得更加简单 */ public interface DrawAble { /** * 绘图方法,接收地图坐标 * @param x 横坐标 * @param y 纵坐标 */ void draw(int x, int y); }
- 定义一系列材质类并实现此绘图接口
- 河流类River
package flyweight.texture; import flyweight.DrawAble; /*** @Description 河流类*/ public class River implements DrawAble {/*** 享元属性: 河流图片材质作为内部属性*/private String image;/*** 类构造器中加载河流图片* 这就是类内部即将共享的“元”数据了,我们通常称之为“内蕴状态”*/public River() {this.image = "河流";System.out.print("从磁盘加载[" + image + "]图片,耗时……");}/*** 重写绘图方法* 而作为“外蕴状态”的坐标是无法作为享元的,由外部传入* @param x 横坐标* @param y 纵坐标*/@Overridepublic void draw(int x, int y) {System.out.println("在位置[" + x + ":" + y + "]上绘制图片:[" + image + "]");} }
- 草地类Grass
package flyweight.texture;import flyweight.DrawAble;/*** @Description 草地类*/public class Grass implements DrawAble {/*** 享元属性:草地图片材质*/private String image;public Grass() {this.image = "草地";System.out.print("从磁盘加载[" + image + "]图片,耗时……");}@Overridepublic void draw(int x, int y) {System.out.println("在位置[" + x + ":" + y + "]上绘制图片:[" + image + "]");}}
- 道路类Road
package flyweight.texture;import flyweight.DrawAble;/*** @Description 道路类*/public class Road implements DrawAble {/*** 道路图片材质*/private String image;public Road() {this.image = "道路";System.out.println("从磁盘加载[" + image + "]图片,耗时…");}@Overridepublic void draw(int x, int y) {System.out.println("在位置[" + x + ":" + y + "]上绘制图片:[" + image + "]");}}
- 房屋类House
package flyweight.texture;import flyweight.DrawAble;/*** @Description 房屋类*/public class House implements DrawAble {private String image;//房屋图片材质public House() {this.image = "房屋";System.out.print("从磁盘加载[" + image + "]图片,耗时……");}@Overridepublic void draw(int x, int y) {System.out.print("将图层切换到顶层……");//房屋盖在地板上,所以切换到顶层图层System.out.println("在位置[" + x + ":" + y + "]上绘制图片:[" + image + "]");}}
- “元之共享”的关键:
- 定义一个图件工厂类
package flyweight.factory;import flyweight.DrawAble;import flyweight.texture.Grass;import flyweight.texture.House;import flyweight.texture.River;import flyweight.texture.Road;import java.util.HashMap;import java.util.Map;/*** @Description 图件工厂类*/public class SegmentFactory {/*** 图库: 维护着所有的图件元对象*/private Map<String, DrawAble> images;public SegmentFactory() {images = new HashMap<String, DrawAble>();}public DrawAble getDrawable(String image) {//缓存池里如果没有图件,则实例化并放入缓存池if(!images.containsKey(image)){switch (image) {case "河流":images.put(image, new River());break;case "草地":images.put(image, new Grass());break;case "道路":images.put(image, new Road());break;case "房屋":images.put(image, new House());}}//至此,缓存池里必然有图件,直接取得并返回return images.get(image);}}
- 并将各种图件对象提前放入内存中共享,如此便可以避免每次从磁盘重新加载
- 测试:Client.test2()
private static void test2() {//先实例化图件工厂SegmentFactory factory = new SegmentFactory();/** 随便绘制一列为例:* 抛弃了利用“new”关键字随意制造对象的方法,* 改用这个图件工厂类来构建并共享图件元,外部需要什么图件直接向图件工厂索要即可*/factory.getDrawable("河流").draw(10, 10);factory.getDrawable("河流").draw(10, 20);factory.getDrawable("道路").draw(10, 30);factory.getDrawable("草地").draw(10, 40);factory.getDrawable("草地").draw(10, 50);factory.getDrawable("草地").draw(10, 60);factory.getDrawable("草地").draw(10, 70);factory.getDrawable("草地").draw(10, 80);factory.getDrawable("道路").draw(10, 90);factory.getDrawable("道路").draw(10, 100);//绘制完地板后接着在顶层绘制房屋factory.getDrawable("房子").draw(10, 10);factory.getDrawable("房子").draw(10, 50);}
- 小结
- 相同部分可以作为享元,如在构造器中加载的,作为内部类即将共享的元数据,通常称为“内蕴状态”
- 不同部分不能作为享元,如在实现房中作为参数传入的属性,称为“外蕴状态”
10.3 总结
- 享元模式让图件对象将可共享的内蕴状态“图片”维护起来,将外蕴状态“坐标”抽离出去并定义于接口参数中
- 基于此,享元工厂便可以顺利将图件对象共享,以供外部随时使用。
- 享元模式的各角色定义如下
- Flyweight(享元接口):所有元件的高层规范,声明与外蕴状态互动的接口标准。如:DrawAble。
- ConcreteFlyweight(享元实现):
- 享元接口的元件实现类,自身维护着内蕴状态,且能接受并响应外蕴状态,
- 可以有多个实现。如:河流类River、草地类Grass、道路类Road等。
- 一个享元对象可以被称作一个“元”
- FlyweightFactory(享元工厂):用来维护享元对象的工厂,负责对享元对象实例进行创建与管理,并对外提供获取享元对象的服务。SegmentFactory
- Client(客户端):享元的使用者,负责维护外蕴状态。Client
- “享元”的理念其实就是萃取事物的本质
- 将对象的内蕴状态与外蕴状态剥离开来,其中内蕴状态成为真正的“元”数据,而外蕴状态则被抽离出去由外部负责维护
相关文章:
秒懂设计模式--学习笔记(11)【结构型-享元模式】
目录 10、享元模式10.1 享元模式10.2 举例10.2.1 马赛克10.2.2 游戏地图(以草原地图作为范例) 10.3 总结 10、享元模式 10.1 享元模式 “享元”则是共享元件的意思享元模式的英文flyweight是轻量级的意思,这就意味着享元模式能使程序变得更…...
Python爬虫——1爬虫基础(一步一步慢慢来)
一、爬虫是什么? (spider) Python 爬虫是利用编程语言 Python 来获取互联网上的数据的技术。它可以自动化地访问网页、提取信息并进行数据处理。以下是Python爬虫的基础知识和步骤: 主要特点和功能: 自动化浏览&#…...
【js自学打卡9】抛出异常 / 幂计算 / 发布订阅 / map小知识点
1. 抛出异常的写法 抛出一个简单的字符串错误 throw Error2; // 抛出一个字符串抛出一个Error对象 throw new Error(出错了!);抛出一个自定义错误对象 function UserError(message) {this.message message;this.name "UserError"; } throw new User…...
ArcGIS Pro SDK (九)几何 7 多点
ArcGIS Pro SDK (九)几何 7 多点 文章目录 ArcGIS Pro SDK (九)几何 7 多点1 构造多点 - 从映射点的枚举2 构造多点 - 使用 MultipointBuilderEx3 修改多点的点4 从多点检索点、2D 坐标、3D 坐标 环境:Visual Studio 2…...
服务器注意事项
1. 远程服务器不允许关机,只能重启; 2. 重启服务器应关闭服务; 3. 不要在服务器访问高峰运行高负载命令; 4. 远程配置防火墙是不要把自己踢出服务器; 5. 制定合理的密码规范并定期更新; 6. 合理分配权…...
学生信息管理系统设计
学生信息管理系统的设计是一个综合性的项目,涉及到数据的存储、检索、更新和删除等基本操作,同时也需要考虑系统的易用性、安全性和扩展性。以下是一些关键步骤和要素,用于指导设计这样一个系统: 1. 需求分析 目标用户ÿ…...
Python求均值,方差,标准差
参考链接:变异系数(Coefficient of Variation,COV)和协方差(Covariance, Cov)-CSDN博客 参考链接:pandas中std和numpy的np.std区别_numpy pandas std-CSDN博客 在计算蛋白质谱数据中的每个蛋白对应的变异…...
DDei在线设计器-HTML渲染
Html渲染 HtmlViewer插件通过将一个外部DIV附着在图形控件上,从而改变原有图形的显示方式。允许使用者自己定义HTML通过HTML元素。本示例演示了通过Html来扩展渲染图形,从而获得更加丰富的图形展现。 通常情况下,我们创建的图形控件ÿ…...
【React Hooks原理 - useSyncExternalStore】
概述 在React项目中说到状态管理,我们第一时间想到的就是使用useState、useReducer这种Hooks来进行状态管理。但是这种是针对React内部的状态,如果有时候我们需要订阅外部的状态并影响React组件的更新的话,那通过这种内部状态管理API显然不能…...
C++STL初阶(7):list的运用与初步了解
在了解了vector之后,我们只需要简单学习List与vector不一样的接口即可 1.list的基本接口 1.1 iterator list中,与vector最大的区别就是迭代器由随机迭代器变成双向迭代器 string和vector中的迭代器都是随机迭代器,支持-等,而LIS…...
el-menu弹出菜单样式不生效
1. 使用 ruoyi 项目时出现的问题。 <template><el-menu:default-active"activeMenu":collapse"false":unique-opened"true"class"container":collapse-transition"true"mode"horizontal"><sideba…...
Springboot 3.x - Reactive programming (2)
三、WebFlux Blocking Web vs. Reactive Web Blocking Web (Servlet) and Reactive Web (WebFlux) have significant differences in several aspects. 1. Front Controller Servlet-Blocking Web: Uses DispatcherServlet as the front controller to handle all HTTP req…...
WPF+Mvvm 项目入门完整教程(一)
WPF+Mvvm 入门完整教程一 创建项目MvvmLight框架安装完善整个项目的目录结构创建自定义的字体资源下载更新和使用字体资源创建项目 打开VS2022,点击创建新项目,选择**WPF应用(.NET Framework)** 创建一个名称为 CommonProject_DeskTop 的项目,如下图所示:MvvmLight框架安装…...
[解决方法]git上传的项目markdown文件的图片无法显示
应该有不少初学者会遇到这种情况 以下是本人摸索出的解决方法 我使用的是typora,首先设置typora的图片设置 文件>偏好设置>图像 如下: 选择这个就会在此文件的同级目录下创建一个assets文件夹来存放此markdown文件的所有图片 然后勾选优先使用相…...
【React】使用 antd 加载组件实现 iframe 的加载效果
文章目录 代码实现: import { Spin } from antd; import { useState } from react;export default function () {const [loading, setLoading] useState(true);return (<div style{{ position: relative, height: 100% }}><Spinstyle{{ position: absolu…...
Python爬虫(1) --基础知识
爬虫 爬虫是什么? spider 是一种模仿浏览器上网过程的一种程序,可以获取一些网页的数据 基础知识 URL 统一资源定位符 uniform resource locator http: 超文本传输协议 HyperText Transfer Protocol 默认端口 80 https: 安全的超文本传输协议 security…...
云原生系列 - Jenkins
Jenkins Jenkins,原名 Hudson,2011 年改为现在的名字。它是一个开源的实现持续集成的软件工具。 官方网站(英文):https://www.jenkins.io/ 官方网站(中文):https://www.jenkins.io…...
django踩坑(四):终端输入脚本可正常执行,而加入crontab中无任何输出
使用crontab执行python脚本时,有时会遇到脚本无法执行的问题。这是因为crontab在执行任务时使用的环境变量与我们在终端中使用的环境变量不同。具体来说,crontab使用的环境变量是非交互式(non-interactive)环境变量,而终端则使用交互式(inter…...
计算机网络入门 -- 常用网络协议
计算机网络入门 – 常用网络协议 1.分类 1.1 模型回顾 计算机网络细分可以划为七层模型,分别是物理层、数据链路层、网络层、传输层、会话层、表示层和应用层。而上三层可以划为应用层中。 1.2 分类 1.2.1 应用层 为用户的应用进程提供网络通信服务࿰…...
【LabVIEW作业篇 - 4】:属性节点赋值和直接节点赋值的区别体现
文章目录 属性节点赋值和直接节点赋值的区别体现 属性节点赋值和直接节点赋值的区别体现 创建5个圆形指示灯,然后循环点亮,先给圆形指示灯赋值假变量,然后再进行循环。 运行结果,观察结果,发现刚开始运行时࿰…...
【数据库系列】Parquet 文件介绍
💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...
A Survey on Multimodal Large Language Models综述
论文题目:A Survey on Multimodal Large Language Models 论文地址:https://arxiv.org/pdf/2306.13549 话题:多模态LLMs综述 MLLMs Paper: https://github.com/BradyFU/Awesome-Multimodal-Large-Language-Models 1. 摘要 近期,以GPT-4V为代表的跨模态大型语言模型(MLLM…...
Leetcode3208. 交替组 II
Every day a Leetcode 题目来源:3208. 交替组 II 解法1:环形数组 把数组复制一份拼接起来,和 3101 题一样,遍历数组的同时,维护以 i 为右端点的交替子数组的长度 cnt。 如果 i ≥ n 且 cnt ≥ k,那么 i…...
汇编教程2
本教程主要教大家如何安装32位Linux虚拟机,为后续实验拆炸弹做准备 下载系统映像文件 以Ubuntu14.04.6系统为例 官方网站:下载地址 点击下载图中32位系统 如果官网进不去可以使用镜像网站 清华镜像网站:下载地址 进入之后找到下图中链接…...
使用 git 和 GitHub 互动
本文根据《GitHub入门与实践》整理 创建账户 要想使用GitHub那就必须先有GitHub账号,账号自行注册,不作介绍。 本地生成 SSH Key SSH 提供了一种安全的方式来通过不安全的网络进行通信。当你使用SSH key连接到GitHub时,你的身份是通过密钥对(一个公钥和一个私钥)来验…...
【Spring Boot 中的 `banner.txt` 和 `logback-spring.xml` 配置】
文章目录 一、banner.txt1. 创建自定义 banner.txt2. 配置 banner.txt 的内容 二、logback-spring.xml1. 创建 logback-spring.xml2. 配置 logback-spring.xml 一、banner.txt banner.txt 是 Spring Boot 项目启动时显示的自定义横幅内容。用来展示项目名称、版本信息或者其他…...
Python Linux环境(Centos8)安装minicoda3+jupyterlab
文章目录 安装miniconda安装python环境启动 最近服务器检查,我下面的服务器有漏洞,不得已重装了,正好记录下怎么从零到python写代码。 安装miniconda miniconda是anconda的精简版,就是管理python环境的得力助手。 # 创建一个名…...
Python PDF Magic:合并和拆分随心所欲
大家好!小编今天要为大家带来一篇关于Python操作PDF的秘籍——无论是要将PDF合并成一份整体,还是将一个庞大的PDF文件拆分成多个小伙伴,都轻松hold住!你准备好了吗?让我们开始这场奇妙的PDF操作之旅吧! 准…...
Gmsh应用程序编程接口
Gmsh应用程序编程接口(API)允许将Gmsh库集成到使用C、C、Python、Julia或Fortran编写的外部应用程序中。从设计上讲,Gmsh API是纯粹功能性的,并且仅使用目标语言的基本类型。 API的结构反映了底层的Gmsh数据模型(也请参…...
DP 203 学习笔记
考试内容总览 Learning Objects: 工具 Designing and implementing data storage 1. Storage Azure Synapse Analytics Azure Databricks Azure Data Lake Storage Gen2(ADLS2,可代替Hadoop Distributed File System也就是HDFS) 2. Shard Partition data store …...
郑州 服装网站建设/做做网站
这个作业属于那个课程 C语言程序设计II 这个作业要求在哪里 https://edu.cnblogs.com/campus/zswxy/computer-scienceclass4-2018/homework/2826 我在这个课程的目标是 熟练指针,能够精通关于数组内部运作原理 这个作业在那个具体方面帮助我实现目标 如何输出…...
网页制作处理中的三剑客/谷歌seo服务
下面列出IE和非IE中常见的一些js兼容性问题。 //window.event IE:有window.event对象 非IE:没有window.event对象。可以通过给函数的参数传递event对象。如οnmοusemοvedoMouseMove(event) 解除冒泡的方法不同 IE:window.event.cancelBubbletr…...
加盟餐饮网站建设/外贸网站哪个比较好
1、修改数据库字符编码 mysql> alter database mydb character set utf8 ; 2、创建数据库时,指定数据库的字符编码 mysql> create database mydb character set utf8 ; 3、查看mysql数据库的字符编码 mysql> show variables like character%; //查询当前my…...
如何做输入密码进入网站/如何快速推广自己的产品
介绍一下Python的数据结构,并说明它们有什么操作 四个基本数据结构:list、tuple、dict、set 在python中,字符串 tuples, 和数字是不可更改的对象,而list,dict等则是可以修改的对象。 1. list 1). list的操作有哪些&…...
网站引导视频怎么做/windows 优化大师
这是一个jsp写的目录查看器转载于:https://blog.51cto.com/chencb19/406596...
企业搭建网站的必要性/广州seo优化推广
安静的早餐Scala学习手册(Learning Scala)第二章1、值与变量定义2、命名3、数据类型及其转换(toType)4、字符串和内插5、元组1、值与变量定义值的定义值,即为常量,不可变,基本语法定义:val : 创建一个名为a,类型为I…...