并发编程(二十三):单例模式(二):静态/非静态方法:单例内存优化关键

单例模式的核心是保证一个类全局仅有一个实例,而实际开发中,我们不仅要实现单例,更要兼顾内存优化、线程安全与扩展性。并发编程(二十二):单例模式总览篇:概念、分类与基础实现-CSDN博客

一、静态方法 vs 非静态方法

"静态方法不用创建对象,为什么不适合作为Servlet/Controller的业务方法?",原因在于内存占用逻辑与扩展性

1 类加载 vs 类初始化

1. 类加载(硬盘 → 内存)

  • 时机 :类首次被使用时(比如第一次调用类的方法、访问类的变量、new 对象),JVM 把硬盘上的 .class 文件加载到内存
  • 内容 :加载类的元信息(类名、方法名、变量名等),但不执行静态代码 / 初始化静态变量

2. 类初始化(内存里初始化)

  • 时机 :类加载后,首次使用类的静态成员(静态方法、静态变量、静态块)时触发
  • 内容 :执行静态块、初始化静态变量,静态成员一旦初始化就永久占内存,直到 JVM 退出

二、静态方法的问题:"不用也占内存"

1. 静态方法的内存逻辑

静态方法 ≠ 不用创建对象,但 类加载 + 初始化后就永久占内存

  • 静态方法属于类本身 ,不是对象,调用时不用 new 对象(比如 ServletUtil.doGet()
  • 只要这个类被加载到内存,静态方法的元信息 + 执行逻辑就永久占内存,哪怕你从没调用过这个静态方法
  • 这和枚举 / 饿汉式单例的问题完全一样:提前占内存,不符合 "按需使用" 的优化

2. 静态方法 vs 非静态方法(单例场景)

特性 静态方法 非静态方法(DCL 单例)
调用方式 类名.方法(不用对象) 对象.方法(需单例对象)
内存占用时机 类加载后就占内存(永久) 第一次调用方法时,创建单例对象才占内存
极限优化适配性 不适配(提前占内存) 适配(按需占内存)
灵活性 差(静态方法不能重写) 好(非静态方法可继承 / 重写)

Q: "为什么不用静态方法代替单例?静态方法不用创建对象啊"

A:

  1. 静态方法虽然不用创建对象,但类一加载,静态方法就占内存------ 哪怕你从没调用过这个静态方法,它也一直占着空间,和枚举的 "提前占内存" 是同一个问题
  2. 而非静态方法(DCL 单例):
    • 类加载时,只加载类的元信息,非静态方法的执行逻辑不占内存
    • 只有第一次调用方法时,才创建单例对象,非静态方法才真正占内存
    • 完全符合 "什么时候用,什么时候才占内存" 的优化需求

简单示例说明:

java 复制代码
// 静态方法版(提前占内存)
class ServletStatic {
    // 类加载后,这个静态方法就占内存,哪怕从没调用过
    public static void doGet() {}
}

// DCL 非静态版(按需占内存)
class ServletDCL {
    private volatile static ServletDCL INSTANCE;
    private ServletDCL() {}

    public static ServletDCL getInstance() {
        if (INSTANCE == null) { // 第一次调用才创建对象
            synchronized (ServletDCL.class) {
                if (INSTANCE == null) {
                    INSTANCE = new ServletDCL(); // 此时非静态方法才占内存
                }
            }
        }
        return INSTANCE;
    }

    // 非静态方法:只有创建对象后才占内存
    public void doGet() {}
}

此外,静态方法无法被重写,灵活性远低于非静态方法,也不符合 Servlet/Controller 可扩展的开发需求。

通过上面的对比我们可以发现,非静态方法必须依托对象才能调用,而想要保证单例的内存优化效果,首先就要从根源上控制对象的创建数量,这就涉及到单例模式最基础的设计:构造方法私有

这就留到下一篇我们再来讲一讲~

并发编程(二十四):单例模式(三):构造方法私有:单例模式的 "第一道防线"-CSDN博客

相关推荐
一叶飘零_sweeeet2 小时前
volatile 关键字深度拆解:从内存屏障底层到单例模式的工业级架构设计
单例模式·volatile
6+h2 小时前
【java IO】缓冲流详解
java·开发语言
爱丽_2 小时前
方法区 / 元空间:JDK 1.7 到 JDK 1.8 到底变了什么?
java·开发语言
JNU freshman2 小时前
Ceph 中与“重构/恢复/回填/重平衡”相关的参数表
java·ceph·重构
寻见9032 小时前
Java为什么能“一次编写,到处运行”?JVM到底解决了什么核心痛点?
java·jvm·java ee
xjdkxnhcoskxbco2 小时前
Java 多线程“八锁”问题深度解析
java·开发语言·多线程
AI克斯2 小时前
【通俗易懂】注解(@)的理解
java
spencer_tseng2 小时前
‘<>‘ operator is not allowed for source level below 1.7
java
人道领域2 小时前
Day | 07 【苍穹外卖:菜品套餐的缓存】
java·开发语言·redis·缓存击穿·springcache