0

0

如何实现Java的享元模式_对象池技术与内存优化实战

P粉602998670

P粉602998670

发布时间:2026-02-27 06:39:21

|

826人浏览过

|

来源于php中文网

原创

如何实现java的享元模式_对象池技术与内存优化实战

享元模式不是对象池,别混用 StringInteger 的缓存机制

Java 里最常被误当作“享元模式”的,其实是 String 常量池和 IntegervalueOf() 缓存。它们是 JVM 层的优化手段,不是你写的享元实现。真要落地享元,核心是:**区分内部状态(共享)和外部状态(不共享),把后者从对象里剥离出来传入方法**。

常见错误现象:
– 写了个 Flyweight 类,但所有字段都塞进构造器,每次 new 都不同实例
– 把用户 ID、时间戳这种明显变化的数据硬塞进享元对象里
– 用 ConcurrentHashMap 存享元,却没控制 key 的生成逻辑,导致缓存爆炸

  • 外部状态必须由客户端持有,并在调用 operation() 时传入(比如 flyweight.operation(context)
  • 内部状态必须是 final、不可变、线程安全的(如字体名、颜色值、图标路径)
  • 享元工厂负责唯一性控制——用 Map<key flyweight></key> + 双重检查锁 or computeIfAbsent() 即可,别自己手写同步块

ObjectPool 来自 Apache Commons Pool,它和享元模式无关但常被一起用

享元解决的是“减少相同对象重复创建”,ObjectPool 解决的是“避免频繁 new/destruct 开销”,目标相似,机制不同。一个管“要不要复用”,一个管“能不能复用”。两者可以共存,但不能替代。

使用场景:
– 数据库连接、HTTP 客户端实例、大尺寸缓冲区(如 ByteBuffer
– 对象构造成本高,且生命周期可控(能 clear/reset)

立即学习Java免费学习笔记(深入)”;

Genspark
Genspark

Genspark 是一款创新的 AI 搜索引擎,致力于提供比传统搜索引擎更高效、准确和无偏见的信息获取方式。

下载
  • 别拿 ObjectPool 去池化 StringInteger —— 它们本身轻量,池化反而增加 GC 压力
  • 务必实现 PooledObjectFactoryvalidateObject()destroyObject(),否则脏对象会泄漏或复用失败
  • 注意 GenericObjectPoolConfigmaxIdleminIdle:设太高吃内存,太低起不到复用效果;生产环境建议 minIdle=0,避免空转占资源

手动实现享元工厂时,key 设计决定内存是否失控

享元对象能不能被复用,全看工厂怎么判断“两个请求是否该返回同一个实例”。key 错了,要么重复创建(浪费),要么误共享(状态污染)。

参数差异:
– 用 new Key(font, size, bold) 比用 font + "|" + size + "|" + bold 更安全(避免特殊字符、null 导致 hash 不一致)
– 如果 key 含业务 ID(如 tenantId),确认该 ID 是否真的属于内部状态——多数情况它属于外部状态,不该进 key

  • 优先用 record(Java 14+)或自定义 equals()/hashCode() 的不可变类作 key,别依赖 toString()
  • 如果 key 组合维度多(比如 5 个字段),考虑用 Objects.hash(a,b,c,d,e) 而不是手拼字符串,减少内存分配
  • 监控 pool.size() 或 map 的 entry 数量,突然增长说明 key 设计有歧义(比如把时间戳当内部状态)

性能陷阱:享元 + 同步 + 外部状态 = 隐形锁竞争

表面上享元省了对象创建,但如果每个 operation() 都要加锁处理外部状态,或者共享对象内部维护了非线程安全的缓存(如 SimpleDateFormat),性能反而比直接 new 更差。

容易踩的坑:
– 在享元里缓存 LocalDateTime.now() 结果,以为省计算,结果时间不准还线程不安全
– 用 ThreadLocal 存外部状态,但没重置,导致内存泄漏
– 享元方法里调用了外部服务(如 DB 查询),把本该并行的请求串行化了

  • 享元对象自身必须无状态或只含不可变状态;所有可变逻辑移到外部或方法参数中
  • 如果 operation 需要临时缓存,用 ThreadLocal<builder></builder> 之类一次性的,用完 remove()
  • 压测时对比 “享元版” 和 “直接 new 版” 的吞吐与 GC pause,尤其关注 Old Gen 占用——享元省的是 young gen 分配,但若 key 泄漏,old gen 会悄悄涨

真正难的不是写工厂或池,是准确识别哪些数据该“内化”,哪些必须“外提”。这个边界划错一次,后面全是并发 bug 和内存泄漏。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

248

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

906

2024.03.01

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1560

2023.10.24

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

638

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

218

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1560

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

643

2023.11.24

Golang 实际项目案例:从需求到上线
Golang 实际项目案例:从需求到上线

《Golang 实际项目案例:从需求到上线》以真实业务场景为主线,完整覆盖需求分析、架构设计、模块拆分、编码实现、性能优化与部署上线全过程,强调工程规范与实践决策,帮助开发者打通从技术实现到系统交付的关键路径,提升独立完成 Go 项目的综合能力。

1

2026.02.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3.9万人学习

C# 教程
C# 教程

共94课时 | 10.2万人学习

Java 教程
Java 教程

共578课时 | 72.7万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号