__init__不是构造函数而是初始化钩子,对象已存在时执行,仅设属性不返回值;它在__new__分配内存后自动调用,self指新实例,参数对应调用值,应避免耗时操作与可变默认参数。

Python类的__init__方法不是构造函数,而是实例创建后的初始化钩子。对象在__init__执行前已经存在,它只负责设置属性、配置状态,不返回值(返回非None会报错)。
__init__ 方法的触发时机
__init__在__new__之后自动调用。__new__负责分配内存并返回新实例,__init__接收这个实例进行初始化。你一般不用重写__new__,除非要控制对象创建过程(比如单例、不可变类型)。
- 写
MyClass()时,Python先调MyClass.__new__(MyClass),再调obj.__init__() - 如果
__init__没定义,父类(通常是object)的空__init__会被调用 - 显式调用
__init__(如obj.__init__())是合法的,但通常没必要,容易重复初始化
参数与self的关系
self是第一个隐式参数,代表刚创建好的实例。它不是关键字,但约定必须叫self。其他参数对应类调用时传入的值。
- 定义:
def __init__(self, name, age): - 调用:
Person("Alice", 30)→self指向新Person对象,name和age被赋值 - 所有实例属性都应通过
self.attr = value方式绑定,否则只是局部变量
常见误用与注意事项
__init__里不适合做耗时操作或外部依赖初始化(如连接数据库),也不该抛出异常以外的逻辑错误——它应专注“让对象进入可用状态”。
立即学习“Python免费学习笔记(深入)”;
- 不要在
__init__中返回值(包括return或return None以外的任何东西) - 避免在
__init__中调用可能失败的I/O操作;考虑用工厂方法或显式load()/connect()方法替代 - 若需默认值,优先用
def __init__(self, items=None):而非def __init__(self, items=[]):(避免可变默认参数陷阱)
与__new__配合的典型场景
当需要干预实例创建本身时,才重写__new__。例如实现单例模式:
class Singleton:
_instance = None
def __new__(cls):
if cls._instance is None:
cls._instance = super().__new__(cls)
return cls._instance
def __init__(self):
if not hasattr(self, 'initialized'): # 防止重复初始化
self.initialized = True
print("Initialized")










