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编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

Elasticsearch 在 Java 中的使用教程

《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("