【高薪程序员必看】万字长文拆解Java并发编程!(8):设计模式-享元模式设计指南

简介: 🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的经典对象复用设计模式-享元模式,废话不多说让我们直接开始。

 

image.gif 编辑

🌟 大家好,我是摘星! 🌟

今天为大家带来的是并发编程中的经典对象复用设计模式-享元模式,废话不多说让我们直接开始。

目录

8. 享元模式

8.1. 享元模式体现

8.1.1. 包装类

8.1.2. 字符串常量池

8.1.3. BigDecimal和BigInteger

8.1.4. 静态工厂

8.2. JVM优化策略


8. 享元模式

享元模式是一种结构型设计模式,它旨在通过共享相同类型且相同值的对象来减少内存消耗和对象创建的开销。在Java中,享元模式的实现涉及两个主要角色:享元工厂和享元对象。

角色

补充说明

示例

享元工厂

应实现为单例模式,使用双重检查线程安全

LongCache

使用静态初始化块保证线程安全

享元对象

必须设计为不可变类,所有属性用final

修饰

Long.valueOf()

返回的对象不可变

外部状态

需通过方法参数传递,不能存储在享元对象中

BigDecimal

的运算结果每次创建新对象

8.1. 享元模式体现

8.1.1. 包装类

在JDK中BooleanByteShortIntegerLongCharacter 等包装类提供了valueOf方法,这个方法就使用到了享元模式

  • 例如Long的valueOf会缓存-128~127之间的Long对象,在这个范围之间会重用对象,大于这个范围,才会新建Long对象
  • ByteShortIntegerLong的缓存范围都是-128~127,Integer的最小值不能变,但是最大值可以调整JVM参数调整
  • Boolean缓存了TRUE和FALSE
  • Character缓存范围是0~127
private static class LongCache {
    private LongCache(){}
    static final Long cache[] = new Long[-(-128) + 127 + 1];
    static {
        for(int i = 0; i < cache.length; i++)
        cache[i] = new Long(i - 128);
    }
}
public static Long valueOf(long l) {
final int offset = 128;
if (l >= -128 && l <= 127) { // will cache
    return LongCache.cache[(int)l + offset];
}
return new Long(l);
}

image.gif

包装类

默认缓存范围

可调整参数

线程安全实现

Integer

-128~127

-XX:AutoBoxCacheMax=250

静态final数组

Long

-128~127

不可调整

类加载时初始化

Character

0~127

不可调整

静态代码块初始化

Boolean

TRUE/FALSE

不可调整

静态final常量

8.1.2. 字符串常量池

// 编译器优化示例
String s1 = "摘星";          // 直接存入常量池
String s2 = new String("摘星").intern(); // 手动入池
// 内存结构图示
┌───────────┐    ┌───────┐
| 堆        | ←──| 引用  |
| String对象|    └───────┘
└───────────┘        ↑
┌───────────┐    ┌───┴───┐
| 方法区     |    | 字面量 |
| 常量池    | ←──| "摘星" |
└───────────┘    └───────┘

image.gif

// 基准测试结果(纳秒/操作)
+-------------------+----------+----------+
| 操作类型          | JDK8     | JDK17    |
+-------------------+----------+----------+
| 常量池查找        | 15       | 12       |
| 新建String对象    | 85       | 78       |
| intern()调用      | 120      | 95       |
+-------------------+----------+----------+

image.gif

8.1.3. BigDecimal和BigInteger

// 反例:BigDecimal未使用享元
BigDecimal a = new BigDecimal("10.00");  // 每次新建对象
BigDecimal b = new BigDecimal("10.00");  // 内存地址不同
// 正例:使用valueOf优化(部分版本缓存0-10)
BigDecimal c = BigDecimal.valueOf(10);   // 可能返回缓存对象

image.gif

8.1.4. 静态工厂

public class Money {
    private static final BigDecimal[] CACHE = new BigDecimal[100];
    static {
        for(int i=0; i<100; i++) {
            CACHE[i] = new BigDecimal(i);
        }
    }
    
    public static BigDecimal valueOf(int val) {
        return (val >=0 && val <100) ? CACHE[val] : new BigDecimal(val);
    }
}
public BigDecimal add(BigDecimal augend) {
    return new BigDecimal(this.value + augend.value); // 伪代码
}

image.gif

8.2. JVM优化策略

逃逸分析:

// 可能被栈上分配的情况(JDK17实测)
void calculate() {
    Long temp = 127L;  // 可能被优化为原始类型
    System.out.println(temp);
}

image.gif

缓存策略

策略类型

优点

缺点

适用场景

预加载缓存(Integer)

无竞争

内存占用固定

值域明确

延迟加载(String.intern)

按需缓存

需同步控制

字符串去重

动态扩展(-XX:AutoBoxCacheMax)

灵活调整

启动后不可变

业务特定范围

目录
相关文章
|
16天前
|
Java 数据库连接 API
2025 更新必看:Java 编程基础入门级超级完整版指南
本教程为2025更新版Java编程基础入门指南,涵盖开发环境搭建(SDKMAN!管理JDK、VS Code配置)、Java 17+新特性(文本块、Switch表达式增强、Record类)、面向对象编程(接口默认方法、抽象类与模板方法)、集合框架深度应用(Stream API高级操作、并发集合)、模式匹配与密封类等。还包括学生成绩管理系统实战项目,涉及Maven构建、Lombok简化代码、JDBC数据库操作及JavaFX界面开发。同时提供JUnit测试、日志框架使用技巧及进阶学习资源推荐,助你掌握Java核心技术并迈向高级开发。
93 5
|
22天前
|
JavaScript 前端开发 Java
Java 编程进阶实操中工具集整合组件封装方法与使用指南详解
本文详细介绍Hutool工具集和图书管理系统相关组件的封装方法及使用示例。通过通用工具类封装(如日期格式化、字符串处理、加密等)、数据库操作封装(结合Hutool DbUtil与MyBatis)、前端Vue组件封装(图书列表与借阅表单)以及后端服务层封装(业务逻辑实现与REST API设计),帮助开发者提升代码复用性与可维护性。同时,提供最佳实践建议,如单一职责原则、高内聚低耦合、参数配置化等,助力高效开发。适用于Java编程进阶学习与实际项目应用。
94 10
|
16天前
|
Oracle Java 关系型数据库
java 编程基础入门级超级完整版教程详解
这份文档是针对Java编程入门学习者的超级完整版教程,涵盖了从环境搭建到实际项目应用的全方位内容。首先介绍了Java的基本概念与开发环境配置方法,随后深入讲解了基础语法、控制流程、面向对象编程的核心思想,并配以具体代码示例。接着探讨了常用类库与API的应用,如字符串操作、集合框架及文件处理等。最后通过一个学生成绩管理系统的实例,帮助读者将理论知识应用于实践。此外,还提供了进阶学习建议,引导学员逐步掌握更复杂的Java技术。适合初学者系统性学习Java编程。资源地址:[点击访问](https://pan.quark.cn/s/14fcf913bae6)。
70 2
|
22天前
|
前端开发 Java 数据库连接
Java 编程进阶实操之工具集整合应用指南
本文聚焦Java编程进阶实操,涵盖并发编程、性能优化及数据库操作优化等核心知识点,并结合Hutool、Postman、Git等实用工具,提供从理论到实践的学习路径。通过小型图书管理系统实战项目,详细解析技术选型与实现步骤,助力开发者掌握Spring Boot、MyBatis等框架应用。同时展望Java新特性与技术趋势,为职业发展奠定基础。资源链接:[点此获取](https://pan.quark.cn/s/14fcf913bae6)。
52 1
|
16天前
|
人工智能 Java API
Java并发编程之Future与FutureTask
本文深入解析了Future接口及其实现类FutureTask的原理与使用。Future接口定义了获取任务结果、取消任务及查询任务状态的规范,而FutureTask作为其核心实现类,结合了Runnable与Future的功能。文章通过分析FutureTask的成员变量、状态流转、关键方法(如run、set、get、cancel等)的源码,展示了异步任务的执行与结果处理机制。最后,通过示例代码演示了FutureTask的简单用法,帮助读者更直观地理解其工作原理。适合希望深入了解Java异步编程机制的开发者阅读。
|
12月前
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
142 4
|
7月前
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
97 4
|
设计模式 安全 Java
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
|
9月前
|
设计模式 安全 Java
Java 编程中的设计模式:单例模式的深度解析
【9月更文挑战第22天】在Java的世界里,单例模式就像是一位老练的舞者,轻盈地穿梭在对象创建的舞台上。它确保了一个类仅有一个实例,并提供全局访问点。这不仅仅是代码优雅的体现,更是资源管理的高手。我们将一起探索单例模式的奥秘,从基础实现到高级应用,再到它与现代Java版本的舞蹈,让我们揭开单例模式的面纱,一探究竟。
82 11
|
9月前
|
设计模式 Java 安全
Java设计模式-单例模式(2)
Java设计模式-单例模式(2)

热门文章

最新文章