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

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

相关推荐
代码AI弗森1 小时前
一文理清楚“算力申请 / 成本测算 / 并发评估”
java·服务器·数据库
Old Uncle Tom2 小时前
OpenClaw 记忆系统 -- 记忆预加载
java·数据结构·算法·agent
小小小米粒2 小时前
Collection单列集合、Map(Key - Value)双列集合,多继承实现。
java·开发语言·windows
摇滚侠2 小时前
expdp 查看帮助
java·数据库·oracle
:1213 小时前
java基础
java·开发语言
曹牧4 小时前
Spring:@RequestMapping注解,匹配的顺序与上下文无关
java·后端·spring
daixin88484 小时前
cursor无法正常使用gpt5.5等模型解决方案
java·redis·cursor
韦禾水5 小时前
记录一次项目部署到tomcat的异常
java·tomcat
曦月合一5 小时前
树莓派安装jdk、tomcat、vnc、谷歌浏览器开机自启等环境配置
java·tomcat·树莓派
此剑之势丶愈斩愈烈5 小时前
openssl 自建证书
java