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

灵活调整

启动后不可变

业务特定范围

目录
相关文章
|
1月前
|
Java
如何在Java中进行多线程编程
Java多线程编程常用方式包括:继承Thread类、实现Runnable接口、Callable接口(可返回结果)及使用线程池。推荐线程池以提升性能,避免频繁创建线程。结合同步与通信机制,可有效管理并发任务。
146 6
|
1月前
|
IDE Java 编译器
java编程最基础学习
Java入门需掌握:环境搭建、基础语法、面向对象、数组集合与异常处理。通过实践编写简单程序,逐步深入学习,打牢编程基础。
200 0
|
2月前
|
SQL Java 数据库
2025 年 Java 从零基础小白到编程高手的详细学习路线攻略
2025年Java学习路线涵盖基础语法、面向对象、数据库、JavaWeb、Spring全家桶、分布式、云原生与高并发技术,结合实战项目与源码分析,助力零基础学员系统掌握Java开发技能,从入门到精通,全面提升竞争力,顺利进阶编程高手。
558 1
|
1月前
|
安全 前端开发 Java
从反射到方法句柄:深入探索Java动态编程的终极解决方案
从反射到方法句柄,Java 动态编程不断演进。方法句柄以强类型、低开销、易优化的特性,解决反射性能差、类型弱、安全性低等问题,结合 `invokedynamic` 成为支撑 Lambda 与动态语言的终极方案。
146 0
|
6月前
|
设计模式 Java 数据库连接
【设计模式】【创建型模式】工厂方法模式(Factory Methods)
一、入门 什么是工厂方法模式? 工厂方法模式(Factory Method Pattern)是一种创建型设计模式,它定义了一个用于创建对象的接口,但由子类决定实例化哪个类。工厂方法模式使类的实例化延迟
198 16
|
6月前
|
设计模式 负载均衡 监控
并发设计模式实战系列(2):领导者/追随者模式
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发设计模式实战系列,第二章领导者/追随者(Leader/Followers)模式,废话不多说直接开始~
204 0
|
6月前
|
设计模式 监控 Java
并发设计模式实战系列(1):半同步/半异步模式
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发设计模式实战系列,第一章半同步/半异步(Half-Sync/Half-Async)模式,废话不多说直接开始~
191 0
|
6月前
|
设计模式 安全 Java
并发设计模式实战系列(12):不变模式(Immutable Object)
🌟 大家好,我是摘星!🌟今天为大家带来的是并发设计模式实战系列,第十二章,废话不多说直接开始~
163 0
|
6月前
|
设计模式 算法 Java
设计模式觉醒系列(04)策略模式|简单工厂模式的升级版
本文介绍了简单工厂模式与策略模式的概念及其融合实践。简单工厂模式用于对象创建,通过隐藏实现细节简化代码;策略模式关注行为封装与切换,支持动态替换算法,增强灵活性。两者结合形成“策略工厂”,既简化对象创建又保持低耦合。文章通过支付案例演示了模式的应用,并强调实际开发中应根据需求选择合适的设计模式,避免生搬硬套。最后推荐了JVM调优、并发编程等技术专题,助力开发者提升技能。
|
11月前
|
设计模式 前端开发 搜索推荐
前端必须掌握的设计模式——模板模式
模板模式(Template Pattern)是一种行为型设计模式,父类定义固定流程和步骤顺序,子类通过继承并重写特定方法实现具体步骤。适用于具有固定结构或流程的场景,如组装汽车、包装礼物等。举例来说,公司年会节目征集时,蜘蛛侠定义了歌曲的四个步骤:前奏、主歌、副歌、结尾。金刚狼和绿巨人根据此模板设计各自的表演内容。通过抽象类定义通用逻辑,子类实现个性化行为,从而减少重复代码。模板模式还支持钩子方法,允许跳过某些步骤,增加灵活性。
638 11

热门文章

最新文章