首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
Python单例模式的几种实现方式

Python单例模式的几种实现方式

热心网友
96
转载
2026-05-01

前言

最近在整理项目代码,又一次审视了单例模式的各种实现。说来有趣,对单例模式的理解,似乎走过了一个完整的循环:从最初懵懂地套用,到后来热衷于追求各种“巧妙”甚至“炫技”的写法,最后兜兜转转,还是觉得最朴素的方案最可靠。

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

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:
    ...

为什么?

  1. 透明性:使用者能一眼看出这个对象是全局共享的,没有隐藏的魔法。
  2. 可控性:创建时机、创建方式完全由你掌控,灵活性更高。
  3. 易测试:在单元测试中,Mock或者替换一个函数返回的实例,比替换一个单例类本身要容易得多。
  4. 少歧义:避免了使用者误以为每次调用构造函数都会得到一个新实例的尴尬。

五、多年编程感悟

朴素的就是最好的。所有当时自鸣得意的所谓编程技巧,最后都会被证明要用更多的代价来理解和维护它。

这句话可能道出了很多资深开发者的心声。追求“巧妙”有时反而会引入不必要的复杂度。

几个教训

当年觉得"巧妙" 后来发现的问题
复杂的继承链 调试时一头雾水,方法调用路径像迷宫
动态修改类属性 程序状态难以追踪,行为不可预测
魔法装饰器 团队新人看得云里雾里, onboarding 成本高
元编程炫技 往往是给自己或后人挖坑, debug 如同解谜

回归朴素

代码的演进,常常是一个从复杂回归简单的过程。

# 曾经的我:一定要用上单例模式这个“高级”特性!
@singleton
class ConfigManager:
    ...

# 现在的我:简单、直接,往往就是最好的
@lru_cache
def get_db():
    return Database()

简单、直白、不需要额外的解释。这样的代码,才是真正的好代码。

六、总结

最后,我们来回顾一下今天的要点。

要点回顾

  1. 装饰器方式:如果你确定某个类必须且永远是单例,那么类装饰器是实现方式中最推荐的一种。
  2. 显式获取单例:在大多数情况下,通过一个函数显式地获取单例实例,比把类本身设计成单例更清晰、更灵活。
  3. lru_cache:对于简单的单例需求,functools.lru_cache 是最轻量、最Pythonic的缓存式解决方案。
  4. 朴素思维:这是最关键的一点。能够用简单方式解决的问题,就不要过度设计。代码的可读性和可维护性,永远应该排在“炫技”的前面。
来源:https://www.jb51.net/python/3631778jr.htm
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

ThinkTask : 聊天式任务管理,提供自动生成报告和任务洞察
AI
ThinkTask : 聊天式任务管理,提供自动生成报告和任务洞察

需求人群 无论是独立工作的个人,还是需要紧密协同的团队,如果你们正在寻找更高效的任务管理与协作方式,那么这款工具很可能就是为你准备的。 产品特色 它的核心能力,可以概括为几个关键的自动化与协同维度。 首先,是自动生成报告和洞察。告别手动整理周报或项目汇总的繁琐,系统能自动梳理进度,提炼关键信息,让你

热心网友
05.02
BoozyBlend : AI定制的最佳鸡尾酒食谱
AI
BoozyBlend : AI定制的最佳鸡尾酒食谱

需求人群 如果你对鸡尾酒感兴趣,无论是专业调酒师还是在家小酌的爱好者,BoozyBlend都能为你提供灵感。这个平台的核心,就是帮你探索新口味、学习调制技巧,并且根据你的独特偏好,创造出专属于你的那一杯。可以说,从入门到精通,它都能全程陪伴。 产品特色 那么,它具体能做什么?亮点主要集中在几个方面:

热心网友
05.02
课灵PPT-免费试用、收费介绍、效果评测、官网入口及在线体验、APP下载和教程
AI
课灵PPT-免费试用、收费介绍、效果评测、官网入口及在线体验、APP下载和教程

课灵PPT 是什么? 说到为教育工作者减负,如今市面上可选的AI工具不少,但能精准切中“课件制作”这个专业需求的,课灵PPT算是一个典型代表。它本质上是一个专为教育场景深度定制的AI智能PPT生成平台。无论是日常教学课件、公开课演示稿,还是家庭辅导材料、儿童启蒙内容,它都能一手包办。 其核心能力在于

热心网友
05.02
Seance AI : AI沟通已故亲友
AI
Seance AI : AI沟通已故亲友

需求人群 当思念无处安放,有些人选择借助科技的力量,延续那份未能尽述的温情。这款工具的核心用户,正是那些渴望与已故亲友进行某种形式沟通的人。它提供了一个私密的空间,让未尽的对话得以继续,让绵长的思念有一个具体的载体。 产品特色 那么,它具体能做什么?关键在于模拟对话体验。用户可以与基于已故亲友信息塑

热心网友
05.02
Nano Banana Pro 图片生成器全面评测|iMini AI 超级智能体-免费试用、收费介绍、效果评测、官网入口及在线体验、APP下载和教程
AI
Nano Banana Pro 图片生成器全面评测|iMini AI 超级智能体-免费试用、收费介绍、效果评测、官网入口及在线体验、APP下载和教程

iMini AI 是什么? 如果说 iMini AI 的“超级 AI 创作系统”是一个强大的创意引擎,那么其中的 Nano Banana Pro,无疑是这个引擎里一颗高性能的核心。它本质上是一个高级的 AI 图像生成器,但定位远超一个简单的文生图工具。通过整合新一代的图像与视频生成模型,再配上庞大的

热心网友
05.02