java设计模式享元模式_Java设计模式百例 - 享元模式

2024-01-17 05:30

本文主要是介绍java设计模式享元模式_Java设计模式百例 - 享元模式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

享元模式(Flyweight Pattern)以共享的方式支持大量的细粒度的对象。尝试重用现有的同类对象,如果未找到匹配的对象,则创建新对象。从而减少创建对象的数量和内存占用量,提高性能。这种类型的设计模式属于结构型模式。

如果说其他的设计模式从名称还能够直观了解到其具体模式或应用场景,那么这个“享元(Flyweight)”。。。好吧,恕我眼拙,无论中文还是英文,确实一眼看不出它是个什么鬼~

先不管它,猜猜下边这个东西是啥,干啥用的~

fe6df87d47b7

flyweight-lifeexample.jpg

想必你已经猜个八九不离十了,这是活字印章,也叫组合印章,可以用在很多需要打印体的纸面材料上面。这可是来自老祖宗的智慧,享元模式可以说跟它有异曲同工之妙。怎么说?

印刷体手写起来很难写,即使能写也好耗费很多时间,用印章模子的方式固定下来,用的时候蘸一下印泥一按就好,省事儿了很多(切!这不是原型模式的思想吗!别急,还没说完~)。如果有人的电话号码是18610861680,那需要做三个1、8和6的印章吗?显然不用,稍微动点脑筋就知道,每次需要印“1”的时候,就用“1”这个印章就好了,其他数字同理。这种对象的复用的思路就跟享元模式比较相似了。

这个例子如果用代码来做如下(本例灵感来自《图解设计模式》的示例程序,以下代码虽同此书不同,但相差不大)。

假设每个印章的印刷体效果如下(在线生成字符字的网站):

fe6df87d47b7

flyweight-font.png

将它们做成10个10×10的字体文件(放到工程项目的resources下),比如fonts/seal-0.txt为

$$$$$$\

$$$ __$$\

$$$$\ $$ |

$$\$$\$$ |

$$ \$$$$ |

$$ |\$$$ |

\$$$$$$ /

\______/

对于数字印章来说,有两个成员变量,一个为数字“0”,还有一个是10行的String[]数组,用来保存以上字体信息。在制作印章(构造方法)的时候就需要指定数字,并从相应的fonts/seal-n.txt中读取字体信息。

NumSeal.java

public class NumSeal {

private Integer num;

private String[] fontLines;

// 构造方法中读取字体信息

public NumSeal(Integer num) {

this.num = num;

this.fontLines = new String[10];

try {

BufferedReader reader = new BufferedReader(new InputStreamReader(getClass().getResourceAsStream("/fonts/seal-" + num + ".txt")));

String line;

for (int i = 0; i < 10; i++) {

line = reader.readLine();

if (line != null) {

fontLines[i] = line;

} else {

break;

}

}

reader.close();

} catch (IOException e) {

e.printStackTrace();

}

}

// 按行打印,共10行10列靠左打印

public void sealPrint(int line) {

System.out.printf(" %-10.10s", fontLines[line]);

}

}

印章工厂,通过维护一个Map,来保证不会重复造出多余的NumSeal对象,实现共享实例的功能,如果没有某个数字的印章,那么现造一个,如果有,就用现有的。

NumSealFactory.java

public class NumSealFactory {

private Map seals = new HashMap(10);

public NumSeal getSeal(Integer num) {

NumSeal seal = seals.get(num);

if (seal == null) {

seal = new NumSeal(num);

System.out.println("制作一个数字为" + num + "的印章");

seals.put(num, seal);

return seal;

}

return seal;

}

}

我们在使用组合印章的时候,给定一个数字字符串,比如手机号“18610861680”,然后使用印章将其打印出来:

NumsPrinter.java

public class NumsPrinter {

private NumSeal[] seals;

public NumsPrinter(String nums) {

seals = new NumSeal[nums.length()];

NumSealFactory numSealFactory = new NumSealFactory();

for (int i = 0; i < nums.length(); i++) {

seals[i] = numSealFactory.getSeal(nums.charAt(i) - 48);

}

}

public void print() {

for (int l = 0; l < 10; l++) {

for (NumSeal seal:seals) {

seal.sealPrint(l);

}

System.out.println();

}

}

}

这里,使用了印章工厂NumSealFactory的实例来获取印章,保证实例共享,并打印出来(由于标准输出玩的不熟,打印的过程是整体按行打印的,跟使用数字印章的时候每个印章打印一下不太一样,不过不影响理解)。

验证一下:

Client.java

public class Client {

public static void main(String[] args) {

new NumsPrinter("18610861680").print();

}

}

输出为:

fe6df87d47b7

flyweight-output.png

可以看到,总共制作了4个印章,重复的数字1个印章就够了。

总结

仍然不给出类关系图和角色说明哈,因为我不是在写教科书。如果你看了这篇文字,之后能够有一个感性的认识,我就倍感荣幸了。还是那句话,设计模式的类关系不是固定的,不在于是用接口还是抽象类做抽象,也不仅仅就是这23个模式,不管黑猫白猫,能抓老鼠就是好猫。设计模式也是同样,世界上本没有设计模式,用的人多了也就有了设计模式,仅此而已。就像上边的例子,NumSealFactory和NumsPrinter如果合并到一个类里边,仍然是享元模式,因为这个模式的特征不在几个类,如何引用的,而在于:

通过一个“对象池”来维护共享的对象,有则拿出来用,无则创建一个放到池子里以便以后也能随时用。如何维护“对象池”呢,通常用Map这种key-value的集合,有一个工厂类来管理,所以享元模式通常也用到了工厂模式。

提到“对象池”,以及上边例子的NumSealFactory,是不是有种似曾相识的感觉。我们在用Hibernate的时候,也有“连接池”,也有“SessionFactory”来获取连接。但是请注意,享元模式的这种对象共享和连接池还不太一样,连接池中的对象都是一致的,取出任何一个连接都可以用来进行CRUD;而享元模式的对象池中的对象是各不相同的,简单理解就是“去重”了,比如上边例子中电话号码”18610861680“去重之后需要”1“、”8“、”6“、”0“四个印章,但是互相之间是不同的”8“的印章印不出6来,如果连接池中的连接都“去重”,那么就剩一个连接了。所以再回想享元模式,关键字可以是”去重对象池“。

好处不多说,去重之后,对象数量少了,更加节省内存;对于对象创建起来比较消耗时间和资源的,重用效率更高;当对象发生变化的时候,变一个就好,比如印章6的字体变了,调整一下这个印章对象就可以了。缺点就是代码结构复杂了,因为需要托管对象创建的过程来进行限制。

其实SDK中,有些不变类(final class)就用到了享元模式的思想,比如:

String就是一个不变类,JVM中维护了一个字符串池,a = "abc"; b = "abc"(基于String.intern()方法查看字符串池中的字符串,若有相同的则无需创建直接返回现有字符串)执行后,a和b使用的是同一个字符串对象(a==b),从而避免创建更多的重复字符串对象,之后用c = new String("abc")时,a和c才不是同一个字符串对象(a!=c);

原生类型的装箱类型(比如Integer)也是不变类,Integer中就也用到”去重对象池“来缓存常用数字的方法:

public static Integer valueOf(int i) {

if (i >= IntegerCache.low && i <= IntegerCache.high)

return IntegerCache.cache[i + (-IntegerCache.low)];

return new Integer(i);

}

low到high通常是-128到127,认为这个区间的Integer对象用的比较多,为避免重复而共享对象:

System.out.println(Integer.valueOf(123) == Integer.valueOf(123)); // true

System.out.println(Integer.valueOf(1234) == Integer.valueOf(1234)); // false

上边无论是String.intern()还是Integer.valueOf(),都相当于享元模式中的那个维护对象池的工厂方法。

这篇关于java设计模式享元模式_Java设计模式百例 - 享元模式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/614992

相关文章

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Java访问修饰符public、private、protected及默认访问权限详解

《Java访问修饰符public、private、protected及默认访问权限详解》:本文主要介绍Java访问修饰符public、private、protected及默认访问权限的相关资料,每... 目录前言1. public 访问修饰符特点:示例:适用场景:2. private 访问修饰符特点:示例:

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

SpringBoot使用Apache Tika检测敏感信息

《SpringBoot使用ApacheTika检测敏感信息》ApacheTika是一个功能强大的内容分析工具,它能够从多种文件格式中提取文本、元数据以及其他结构化信息,下面我们来看看如何使用Ap... 目录Tika 主要特性1. 多格式支持2. 自动文件类型检测3. 文本和元数据提取4. 支持 OCR(光学

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.