Python单例模式的几种实现方式
前言
最近在整理项目代码,又一次审视了单例模式的各种实现。说来有趣,对单例模式的理解,似乎走过了一个完整的循环:从最初懵懂地套用,到后来热衷于追求各种“巧妙”甚至“炫技”的写法,最后兜兜转转,还是觉得最朴素的方案最可靠。
免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

今天,我们就来系统梳理一下Python中实现单例的几种主流方式,并分享一个在编程圈子里逐渐形成的共识:很多时候,朴素的就是最好的。
一、单例模式简介
单例模式(Singleton Pattern),这个设计模式的核心目标很明确:确保一个类只有一个实例,并且提供一个全局的访问点。听起来简单,但用起来却有不少门道。
适用场景
| 场景 | 说明 |
|---|---|
| 数据库连接 | 全局共享一个连接池,避免重复建立连接的开销 |
| 日志记录器 | 确保所有日志写入同一个文件或流,避免重复打开文件 |
| 配置管理器 | 全局只读一份配置,保证数据一致性 |
| 缓存实例 | 全局共享缓存,提升访问效率 |
二、四种实现方式
那么,在Python的世界里,有哪些方法可以实现单例呢?下面这四种,算是比较有代表性的。
方式一:类装饰器
def singleton(cls):
"""单例装饰器:闭包保存唯一实例"""
_instance = None
def wrapper(*args, **kwargs):
nonlocal _instance
if _instance is None:
_instance = cls(*args, **kwargs)
return _instance
return wrapper
@singleton
class DBConnect:
def __init__(self, host):
self.host = host
print("数据库连接初始化")
# 使用
obj1 = DBConnect("127.0.0.1")
obj2 = DBConnect("192.168.1.1") # 不会重新初始化
print(obj1 is obj2) # True
print(obj1.host) # 127.0.0.1
特点:装饰器的语法非常优雅,使用时一目了然,算是Pythonic的实现之一。
方式二:__new__方法
class MySingleton:
"""单例类:通过 __new__ 控制实例创建"""
_instance = None
def __new__(cls):
if cls._instance is None:
cls._instance = super().__new__(cls)
return cls._instance
# 使用
obj1 = MySingleton()
obj2 = MySingleton()
print(obj1 is obj2) # True
特点:这是最直接、教科书式的方式。但有个小问题,_instance作为类变量暴露在外。话说回来,之前在一篇关于Playwright封装的文章里就用了这种方式,现在回头看觉得不太妥——那个类本身并不必须是个单例。更好的做法是,把单例化的选择权交给使用者,让他们根据需要显式处理。
方式三:函数
from typing import Type, TypeVar
T = TypeVar('T')
def singleton(cls: Type[T], *args, **kwargs) -> T:
"""单例工厂函数"""
if not hasattr(cls, '_instance'):
cls._instance = cls(*args, **kwargs)
return cls._instance
# 使用
class TargetClass:
def __init__(self, config):
self.config = config
obj = singleton(TargetClass, config="value")
特点:需要显式调用singleton()函数。优点是语义非常明确,使用者清楚地知道自己在获取一个单例;缺点嘛,就是每次都要多写几个字符,不够直观。
方式四:functools.lru_cache
from functools import lru_cache
@lru_cache(maxsize=1)
def get_instance():
return MyClass(arg1="value")
# 使用
obj1 = get_instance()
obj2 = get_instance()
print(obj1 is obj2) # True
特点:直接利用Python标准库,无需额外造轮子。虽然从语义上说它更像一个缓存,但实现单例的效果是杠杠的,而且还能享受缓存机制带来的其他好处。
三、优缺点对比
光看代码可能还不够直观,我们把几种方式放到一起对比一下,优劣就清晰了。
| 实现方式 | 优点 | 缺点 | 推荐指数 |
|---|---|---|---|
| 类装饰器 | 语法优雅、语义清晰、易于理解 | 对于非必要实现单例的类,对使用者会造成困扰 | ⭐⭐⭐⭐ |
| new 方法 | 最直观、最简单 | 实例变量暴露、可复用性不好 | ⭐⭐ |
| 单例函数 | 显式调用、可控性强,使用者清楚知道正在做什么 | 每次都要调用、不够直观 | ⭐⭐⭐ |
| lru_cache | 内置、无需额外代码 | 语义上不是单例、更像缓存,但是胜在简单 | ⭐⭐⭐⭐ |
四、我的建议
看了这么多实现,到底该怎么选呢?这里有一个核心建议。
优先使用显式单例
相比费尽心思把一个类本身变成单例,显式地获取单例往往是更清晰的写法。来看两个例子:
# ✅ 推荐:显式获取单例
_db = None
def get_db():
global _db
if _db is None:
_db = Database()
return _db
# 或者更简洁地用 lru_cache
@lru_cache
def get_db():
return Database()
# ❌ 不推荐:把类变成单例,除非很确定此类必须是单例类,不会造成使用者误解
@singleton
class Database:
...
为什么?
- 透明性:使用者能一眼看出这个对象是全局共享的,没有隐藏的魔法。
- 可控性:创建时机、创建方式完全由你掌控,灵活性更高。
- 易测试:在单元测试中,Mock或者替换一个函数返回的实例,比替换一个单例类本身要容易得多。
- 少歧义:避免了使用者误以为每次调用构造函数都会得到一个新实例的尴尬。
五、多年编程感悟
朴素的就是最好的。所有当时自鸣得意的所谓编程技巧,最后都会被证明要用更多的代价来理解和维护它。
这句话可能道出了很多资深开发者的心声。追求“巧妙”有时反而会引入不必要的复杂度。
几个教训
| 当年觉得"巧妙" | 后来发现的问题 |
|---|---|
| 复杂的继承链 | 调试时一头雾水,方法调用路径像迷宫 |
| 动态修改类属性 | 程序状态难以追踪,行为不可预测 |
| 魔法装饰器 | 团队新人看得云里雾里, onboarding 成本高 |
| 元编程炫技 | 往往是给自己或后人挖坑, debug 如同解谜 |
回归朴素
代码的演进,常常是一个从复杂回归简单的过程。
# 曾经的我:一定要用上单例模式这个“高级”特性!
@singleton
class ConfigManager:
...
# 现在的我:简单、直接,往往就是最好的
@lru_cache
def get_db():
return Database()
简单、直白、不需要额外的解释。这样的代码,才是真正的好代码。
六、总结
最后,我们来回顾一下今天的要点。
要点回顾
- 装饰器方式:如果你确定某个类必须且永远是单例,那么类装饰器是实现方式中最推荐的一种。
- 显式获取单例:在大多数情况下,通过一个函数显式地获取单例实例,比把类本身设计成单例更清晰、更灵活。
- lru_cache:对于简单的单例需求,
functools.lru_cache是最轻量、最Pythonic的缓存式解决方案。 - 朴素思维:这是最关键的一点。能够用简单方式解决的问题,就不要过度设计。代码的可读性和可维护性,永远应该排在“炫技”的前面。
游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。
同类文章
如何优化Apache2响应速度
Apache2响应速度优化实操指南 想让你的Apache2服务器跑得更快?这事儿其实有章可循。下面这份实操指南,将从基础到进阶,帮你系统地提升响应速度。记住,所有优化都建立在不变动核心业务逻辑和架构的前提下。 一 基础与系统层面优化 优化得从地基开始。系统层面的几个关键设置,往往能以小成本换来大收益
git多人协作的工作流程【汇总】
多人协作必须禁用直接 push 到 main 分支:PR MR 流程是保障代码质量、自动化测试与冲突预判的核心机制;最佳实践包括语义化分支命名、启用分支保护规则,并规范 rebase 与 merge 的使用场景。 多人协作时,为什么禁止直接 push 到 main 分支? 直接向主分支推送代码,表面
CentOS上如何升级PHPStorm到最新版本
在 CentOS 上升级 PhpStorm 的可选方案 说到在 CentOS 上升级 PhpStorm,其实路径很清晰。核心原则是:优先使用内置更新或 JetBrains Toolbox App 这类自动管理工具,其次才是手动下载安装包覆盖升级。下面,就按推荐顺序,把每种方式的操作步骤和关键要点给你
Atom如何设置自动保存?Atom自动保存功能开启教程
Atom如何设置自动保存?Atom自动保存功能开启教程 如果你还在为Atom的自动保存功能头疼,那很可能踩中了几个常见的“坑”。从1 27版本开始,autosa ve功能已经作为核心特性内置,不再依赖插件。但问题也随之而来:为什么设置了却不见效?答案往往藏在版本、配置层级,或者那些本该被清理的旧插件
如何在CentOS上备份PHPStorm的配置文件
在 CentOS 上备份 PhpStorm 配置文件:完整指南与最佳实践 一、备份前的准备工作 在开始备份 PhpStorm 配置之前,充分的准备工作至关重要。这能有效保障备份数据的完整性与安全性,避免因操作不当导致配置丢失或损坏。 彻底关闭 PhpStorm 应用程序:这是首要且必须的步骤。确保
- 日榜
- 周榜
- 月榜
1
2
3
4
5
6
7
8
9
10
相关攻略
2015-03-10 11:25
2015-03-10 11:05
2021-08-04 13:30
2015-03-10 11:22
2015-03-10 12:39
2022-05-16 18:57
2025-05-23 13:43
2025-05-23 14:01
热门教程
- 游戏攻略
- 安卓教程
- 苹果教程
- 电脑教程
热门话题

