十一、享元模式

  • 目的 : 运用共享技术有效地支持大量细粒度的对象。
  • 核心 : 区分对象的内部状态 (可共享的、不变的)和外部状态(不可共享的、变化的)。创建享元工厂管理共享的内部状态,客户端使用时传入外部状态。
  • 场景 : 游戏中有大量相同类型的对象(如子弹、树木、士兵)、文本编辑器需要渲染大量字符(如字母、数字、符号)等。

首先有一个享元对象类

java 复制代码
package flyweight;

public class Letter {
	private String name;

	public Letter(String name){
		this.name = name;
	}

	public String getName(){
		return this.name;
	}
}

然后是享元工厂类

java 复制代码
package flyweight;

import java.util.HashMap;
import java.util.Map;

public class LetterFactory {
	private Map<String, Letter> map;	// 对象池
	private static LetterFactory instance = new LetterFactory();  // 1 私有静态实例 instance,通过静态方法 getInstance() 获取。

	private LetterFactory(){
		map = new HashMap<>();	//  2  私有构造函数,禁止外部直接实例化。
	}

	public static  LetterFactory getInstance(){		//  3  确保 LetterFactory 全局只有一个实例。
		return instance;
	}

	public synchronized void add(Letter letter){	 // 确保同名 Letter 只保存一份
		if(letter != null && !map.containsKey(letter.getName())){
			map.put(letter.getName(), letter);		 //	 存入缓存
		}

		System.out.println("map.size ==== " + map.size()); //通过对象池大小检验:是否同名 Letter 只保存一份
	}

	public Letter get(String name){
		return map.get(name);			// 从缓存中获取对象,避免重复创建
	}
}

函数入口

java 复制代码
package flyweight;

public class MainTest {
	public static void main(String[] args) {
		LetterFactory factory = LetterFactory.getInstance();	// 获取单例
		String word = "success";

		addLetterByName(factory, word);
		getLetter(factory, word);
	}

	static void addLetterByName(LetterFactory factory, String word) {
		for (char c : word.toCharArray()) {						// 遍历字符串的每个字符
			factory.add(new Letter(c + ""));					// 创建 Letter 对象并存入工厂,c + "" 的作用是将 char 类型转换为 String 类型
		}
	}

	static void getLetter(LetterFactory factory, String word) {
		for (char c : word.toCharArray()) {
			System.out.println(factory.get(c + ""));			// 从缓存中获取对应的 对象并打印
		}
	}
}

终端输出:

复制代码
map.size ==== 1  // 添加 s
map.size ==== 2  // 添加 u  
map.size ==== 3  // 添加 c
map.size ==== 3  // 第二个 c,已存在,size不变
map.size ==== 4  // 添加 e
map.size ==== 4  // 第二个 s,已存在,size不变  
map.size ==== 4  // 第三个 s,已存在,size不变

flyweight.Letter@4aa298b7  // s
flyweight.Letter@7d4991ad  // u
flyweight.Letter@28d93b30  // c
flyweight.Letter@28d93b30  // c (复用)
flyweight.Letter@1b6d3586  // e  
flyweight.Letter@4aa298b7  // s (复用)
flyweight.Letter@4aa298b7  // s (复用)

LetterFactory 和 Letter 是聚合关系,Letter 可以独立存在;

LetterFactory 和 map 是组合关系,map 是 LetterFactory 的组成部分。

相关推荐
卓怡学长2 小时前
基于 SpringBoot 的生活信息分享平台,从 0 到 1 完整实现(附源码 + 数据库)
java·数据库·spring boot·tomcat·maven
ID_180079054732 小时前
Python解析小红书(XHS)笔记评论 API,json数据返回参考
java·服务器·数据库
努力努力再努力wz2 小时前
【C++高阶系列】告别内查找局限:基于磁盘 I/O 视角的 B 树深度剖析与 C++ 泛型实现!(附B树实现源码)
java·linux·开发语言·数据结构·c++·b树·算法
hero.fei2 小时前
RoaringBitmap在SpringBoot中的使用以及与BitSet对比
java·spring boot·spring
Traving Yu2 小时前
Spring源码与框架原理
java·后端·spring
Lyyaoo.2 小时前
【JAVA基础面经】线程安全的单例模式
java·安全·单例模式
_李小白2 小时前
【OSG学习笔记】Day 39: NodeCallback(帧回调机制)
java·笔记·学习
如来神掌十八式2 小时前
设计模式之装饰器模式
java·设计模式
cch89182 小时前
C++、Python与汇编语言终极对比
java·开发语言·jvm