0

0

Singleton 模式在单线程环境下的正确实现与常见误区

花韻仙語

花韻仙語

发布时间:2025-12-30 13:35:27

|

711人浏览过

|

来源于php中文网

原创

Singleton 模式在单线程环境下的正确实现与常见误区

本文详解单线程环境下 singleton 模式的本质要求:唯一实例 + 禁止外部构造,指出仅靠 `getinstance()` 重赋值无法保证单例性,并对比分析两种写法的合规性。

Singleton(单例)模式的核心契约是:在整个 JVM 生命周期中,该类有且仅有一个实例存在,并且该实例的创建与访问必须受控。这一约束在单线程环境中虽无需考虑并发安全,但依然严格要求两点:

  1. 私有化构造器(private constructor)——彻底阻止外部通过 new 关键字创建新实例;
  2. 全局唯一访问入口(如静态 getInstance() 方法)——确保所有获取实例的路径都指向同一对象。

我们来逐个分析您提供的两个实现:

第一种写法(合规的单例)

public class Singleton {
    private static Singleton obj;

    private Singleton() {} // ✅ 私有构造器:外部无法 new

    public static Singleton getInstance() {
        if (obj == null) {
            obj = new Singleton(); // ✅ 唯一合法的实例创建点
        }
        return obj;
    }
}

该实现满足单例全部前提:构造器私有 → 无外部实例化可能;getInstance() 是唯一获取途径 → 所有调用均返回同一 obj 引用。在单线程下,它能稳定保证全局唯一实例。

第二种写法(非单例,本质失效)

public class SingleTon {
    static SingleTon s;

    SingleTon getInstance() { // ⚠️ 非静态方法,设计混乱
        if (s == null) {
            s = new SingleTon(); // ✅ 第一次创建
        }
        return s;
    }

    public static void main(String[] args) {
        SingleTon s1 = new SingleTon(); // ❌ 公共默认构造器允许直接 new!
        s1 = s1.getInstance();         // ✅ 此时 s1 指向共享实例 s
        SingleTon s2 = new SingleTon(); // ❌ 再次 new —— 创建了第二个独立实例!
        s2 = s2.getInstance();         // ✅ s2 也指向 s,但原 s2 对象已丢失引用(仍存在于内存)
    }
}

问题关键在于:未声明 private SingleTon() → 编译器自动注入 public SingleTon() → 任何代码均可调用 new SingleTon() 创建任意数量实例。即使后续 s1 = s1.getInstance() 将引用重定向到共享对象 s,最初的 new SingleTon() 实例依然真实存在、占用内存、且完全脱离单例管理。这直接违背“有且仅有一个实例”的根本定义。

Fireflies.ai
Fireflies.ai

自动化会议记录和笔记工具,可以帮助你的团队记录、转录、搜索和分析语音对话。

下载

此外,该实现还存在设计缺陷:

  • getInstance() 是实例方法而非 static,导致调用前必须先创建一个临时实例(如 new SingleTon().getInstance()),逻辑冗余且易引发误解;
  • 静态字段 s 与实例方法混用,职责不清,破坏封装性

? 总结与最佳实践

  • 单例的“单”体现在实例数量为一,而非“变量引用指向同一对象”;
  • private 构造器是强制性前提,不可省略或妥协;
  • getInstance() 必须为 public static 方法,确保统一、无依赖的访问入口;
  • 单线程下推荐使用饿汉式(类加载时初始化)或懒汉式(如题中第一种),二者均需私有构造器支撑。

正确实现示例(懒汉式,单线程安全):

public class Singleton {
    private static Singleton instance;

    private Singleton() {} // ? 核心防线:禁止外部构造

    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

牢记:没有私有构造器的“单例”,只是披着单例外衣的普通类。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

467

2023.08.10

excel制作动态图表教程
excel制作动态图表教程

本专题整合了excel制作动态图表相关教程,阅读专题下面的文章了解更多详细教程。

24

2025.12.29

freeok看剧入口合集
freeok看剧入口合集

本专题整合了freeok看剧入口网址,阅读下面的文章了解更多网址。

74

2025.12.29

俄罗斯搜索引擎Yandex最新官方入口网址
俄罗斯搜索引擎Yandex最新官方入口网址

Yandex官方入口网址是https://yandex.com;用户可通过网页端直连或移动端浏览器直接访问,无需登录即可使用搜索、图片、新闻、地图等全部基础功能,并支持多语种检索与静态资源精准筛选。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

207

2025.12.29

python中def的用法大全
python中def的用法大全

def关键字用于在Python中定义函数。其基本语法包括函数名、参数列表、文档字符串和返回值。使用def可以定义无参数、单参数、多参数、默认参数和可变参数的函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

16

2025.12.29

python改成中文版教程大全
python改成中文版教程大全

Python界面可通过以下方法改为中文版:修改系统语言环境:更改系统语言为“中文(简体)”。使用 IDE 修改:在 PyCharm 等 IDE 中更改语言设置为“中文”。使用 IDLE 修改:在 IDLE 中修改语言为“Chinese”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

18

2025.12.29

C++的Top K问题怎么解决
C++的Top K问题怎么解决

TopK问题可通过优先队列、partial_sort和nth_element解决:优先队列维护大小为K的堆,适合流式数据;partial_sort对前K个元素排序,适用于需有序结果且K较小的场景;nth_element基于快速选择,平均时间复杂度O(n),效率最高但不保证前K内部有序。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

12

2025.12.29

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

136

2025.12.29

抖音网页版入口在哪(最新版)
抖音网页版入口在哪(最新版)

抖音网页版可通过官网https://www.douyin.com进入,打开浏览器输入网址后,可选择扫码或账号登录,登录后同步移动端数据,未登录仅可浏览部分推荐内容。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

66

2025.12.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号