【高薪程序员必看】万字长文拆解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)

灵活调整

启动后不可变

业务特定范围

相关文章
|
2月前
|
设计模式 Java Spring
Java 设计模式之责任链模式:优雅处理请求的艺术
责任链模式通过构建处理者链,使请求沿链传递直至被处理,实现发送者与接收者的解耦。适用于审批流程、日志处理等多级处理场景,提升系统灵活性与可扩展性。
314 2
|
2月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
341 0
|
4月前
|
设计模式 缓存 Java
Java设计模式(二):观察者模式与装饰器模式
本文深入讲解观察者模式与装饰器模式的核心概念及实现方式,涵盖从基础理论到实战应用的全面内容。观察者模式实现对象间松耦合通信,适用于事件通知机制;装饰器模式通过组合方式动态扩展对象功能,避免子类爆炸。文章通过Java示例展示两者在GUI、IO流、Web中间件等场景的应用,并提供常见陷阱与面试高频问题解析,助你写出灵活、可维护的代码。
|
5月前
|
人工智能 Cloud Native Java
2025 年 Java 应届生斩获高薪需掌握的技术实操指南与实战要点解析
本指南为2025年Java应届生打造,涵盖JVM调优、响应式编程、云原生、微服务、实时计算与AI部署等前沿技术,结合电商、数据处理等真实场景,提供可落地的技术实操方案,助力掌握高薪开发技能。
270 2
|
2月前
|
设计模式 算法 搜索推荐
Java 设计模式之策略模式:灵活切换算法的艺术
策略模式通过封装不同算法并实现灵活切换,将算法与使用解耦。以支付为例,微信、支付宝等支付方式作为独立策略,购物车根据选择调用对应支付逻辑,提升代码可维护性与扩展性,避免冗长条件判断,符合开闭原则。
387 35
|
7月前
|
人工智能 Kubernetes Java
回归开源,两位 Java 和 Go 程序员分享的开源贡献指引
Higress是一个基于Istio和Envoy的云原生API网关,支持AI功能扩展。它通过Go/Rust/JS编写的Wasm插件提供可扩展架构,并包含Node和Java的console模块。Higress起源于阿里巴巴,解决了Tengine配置重载及gRPC/Dubbo负载均衡问题,现已成为阿里云API网关的基础。本文介绍Higress的基本架构、功能(如AI网关、API管理、Ingress流量网关等)、部署方式以及如何参与开源贡献。此外,还提供了有效的开源贡献指南和社区交流信息。
716 33
|
2月前
|
设计模式 消息中间件 传感器
Java 设计模式之观察者模式:构建松耦合的事件响应系统
观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。
301 8
|
7月前
|
Java 程序员 应用服务中间件
【高薪程序员必看】万字长文拆解Java并发编程!(2 2-2)
📌 核心痛点暴击:1️⃣ 面了8家都被问synchronized锁升级?一张图看懂偏向锁→重量级锁全过程!2️⃣ 线程池参数不会配?高并发场景下这些参数调优救了项目组命!3️⃣ volatile双重检测单例模式到底安不安全?99%人踩过的内存可见性大坑!💡 独家亮点抢先看:✅ 图解JVM内存模型(JMM)三大特性,看完再也不怕指令重排序✅ 手撕ReentrantLock源码,AQS队列同步器实现原理大揭秘✅ 全网最细线程状态转换图(附6种状态转换触发条件表)
135 0
|
4月前
|
设计模式 安全 Java
Java设计模式(一):单例模式与工厂模式
本文详解单例模式与工厂模式的核心实现及应用,涵盖饿汉式、懒汉式、双重检查锁、工厂方法、抽象工厂等设计模式,并结合数据库连接池与支付系统实战案例,助你掌握设计模式精髓,提升代码专业性与可维护性。
|
4月前
|
设计模式 XML 安全
Java枚举(Enum)与设计模式应用
Java枚举不仅是类型安全的常量,还具备面向对象能力,可添加属性与方法,实现接口。通过枚举能优雅实现单例、策略、状态等设计模式,具备线程安全、序列化安全等特性,是编写高效、安全代码的利器。

热门文章

最新文章