0

0

深入理解Python中__new__与__init__方法的执行机制与陷阱

聖光之護

聖光之護

发布时间:2025-08-06 23:02:14

|

982人浏览过

|

来源于php中文网

原创

深入理解python中__new__与__init__方法的执行机制与陷阱

Python中的__new__和__init__方法分别负责对象的创建和初始化。__new__在__init__之前被调用,并负责返回一个实例。本文通过实例代码详细解析了这两个方法的执行顺序,特别是在继承场景下,以及当__new__方法不正确返回实例时可能导致的问题,并指出了避免在__new__中手动调用__init__的常见反模式。

1. __new__与__init__的核心区别

在Python中,对象的生命周期涉及两个关键的特殊方法:__new__和__init__。理解它们的职责和执行顺序对于编写健壮的面向对象代码至关重要。

  • *`new(cls, args, kwargs)`: 这是一个类方法,负责创建并返回一个新实例。它在对象实例化(即调用类名创建对象)时第一个被调用。__new__的典型用途包括实现单例模式、创建不可变对象或控制实例的创建过程。它的第一个参数是cls,表示要实例化的类。
  • *`init(self, args, kwargs)`: 这是一个实例方法,负责初始化一个已经创建好的实例。它在__new__方法返回实例后被调用,并接收该实例作为其第一个参数self。__init__的主要职责是设置实例的属性。

执行顺序总结: 当你通过 ClassName() 创建一个对象时,Python解释器会首先调用 ClassName.__new__() 来创建实例。如果 __new__ 方法成功返回了一个该类的实例(或其子类的实例),那么接下来解释器会自动调用该实例的 __init__() 方法进行初始化。

2. 案例分析:非标准用法导致的执行流

为了更好地理解__new__和__init__的交互,特别是当它们被非标准地实现时,我们来看一个具体的例子。

class Demo:
    def __new__(self): # 注意:这里缺少 cls 参数,且未返回实例
        self.__init__(self) # 反模式:在 __new__ 中显式调用 __init__
        print("Demo's __new__() invoked")
    def __init__(self):
        print("Demo's __init__() invoked")

class Derived_Demo(Demo):
    def __new__(self): # 注意:这里缺少 cls 参数,且未返回实例
        print("Derived_Demo's __new__() invoked")
    def __init__(self):
        print("Derived_Demo's __init__() invoked")

def main():
    obj1 = Derived_Demo()
    obj2 = Demo()
    print(f"obj1 is: {obj1}") # 观察 obj1 的值
    print(f"obj2 is: {obj2}") # 观察 obj2 的值

main()

运行上述代码,输出如下:

立即学习Python免费学习笔记(深入)”;

Derived_Demo's __new__() invoked
Demo's __init__() invoked
Demo's __new__() invoked
obj1 is: None
obj2 is: None

执行流解析:

  1. obj1 = Derived_Demo() 的创建过程:

    万兴爱画
    万兴爱画

    万兴爱画AI绘画生成工具

    下载
    • 首先调用 Derived_Demo.__new__(Derived_Demo)。
    • 打印 "Derived_Demo's __new__() invoked"。
    • 关键点: Derived_Demo.__new__ 方法没有显式地返回任何对象。在Python中,如果一个函数或方法没有显式地 return 语句,它会隐式地返回 None。因此,Derived_Demo.__new__ 返回了 None。
    • 结果: 由于 __new__ 方法没有返回一个有效的实例(而是返回了 None),Python解释器不会再自动调用 Derived_Demo.__init__ 方法。obj1 最终被赋值为 None。
  2. obj2 = Demo() 的创建过程:

    • 首先调用 Demo.__new__(Demo)。
    • 在 Demo.__new__ 内部,有一行 self.__init__(self)。这里的 self 实际上是类 Demo。因此,这行代码显式地调用了 Demo.__init__(Demo)。
    • 打印 "Demo's __init__() invoked"。
    • 接着,Demo.__new__ 继续执行,打印 "Demo's __new__() invoked"。
    • 关键点: 类似于 Derived_Demo.__new__,Demo.__new__ 方法也没有显式地返回任何对象,因此它也隐式地返回了 None。
    • 结果: Python解释器在 __new__ 返回后,不会再自动调用 Demo.__init__(即使它在 __new__ 内部被手动调用过)。obj2 最终被赋值为 None。

从输出结果可以看出,obj1 和 obj2 都变成了 None,这与我们通常创建对象的预期行为大相径庭。

3. 正确理解__new__方法的约定

一个设计良好的__new__方法应遵循以下约定:

  • 正确的签名: __new__方法必须以cls作为其第一个参数,通常还会接受*args和**kwargs以便将参数传递给__init__。
    def __new__(cls, *args, **kwargs):
        # ...
  • 返回实例: __new__方法的核心职责是创建并返回一个类的实例。最常见的做法是调用父类的__new__方法来获取实例。
    class MyClass:
        def __new__(cls, *args, **kwargs):
            instance = super().__new__(cls) # 创建实例
            return instance # 返回实例
  • __init__的自动调用: 只有当__new__方法返回了一个属于当前类或其子类的实例时,Python解释器才会自动调用该实例的__init__方法。如果__new__返回了None、一个非当前类的实例,或者其他非实例值,那么__init__将不会被自动调用。

4. 避免常见反模式:在__new__中调用__init__

在上述案例中,Demo.__new__中显式调用了self.__init__(self)。这是一个应该避免的反模式

为什么是反模式?

  1. 职责混淆: __new__的职责是创建对象,__init__的职责是初始化对象。将初始化逻辑放在__new__中混淆了这两个阶段的职责。
  2. 潜在的重复初始化: 如果__new__正确地返回了一个实例,并且该实例的__init__方法随后被Python自动调用,那么在__new__中手动调用的__init__会导致重复初始化。这可能导致难以调试的副作用或状态错误。
  3. 参数传递问题: __new__和__init__的参数通常是一致的,但它们的处理方式不同。在__new__中手动调用__init__时,你需要手动处理参数传递,这容易出错。
  4. 违反直觉: Python的面向对象模型中,__new__和__init__是解耦的。手动干预这种解耦会使代码更难理解和维护。

5. 总结与最佳实践

  • __new__用于创建实例,__init__用于初始化实例。 这是理解二者关系的核心。
  • __new__总是先于__init__执行。
  • __new__必须返回一个实例。 如果__new__返回None或一个非实例对象,那么__init__将不会被自动调用,你得到的对象将是None或其他非预期的结果。
  • 避免在__new__中手动调用__init__。 这是Python对象创建和初始化流程的反模式,可能导致重复初始化和逻辑混乱。
  • 何时使用__new__: 只有当你需要控制实例的创建过程时(例如,实现单例模式、创建不可变对象、或者需要返回一个不同类型的实例)才应该重写__new__。
  • 何时使用__init__: 在绝大多数情况下,你只需要重写__init__来设置新创建实例的属性。

通过遵循这些原则,你可以更有效地利用Python的面向对象特性,并避免在对象创建和初始化过程中常见的陷阱。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

52

2025.11.27

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

52

2025.11.27

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

52

2025.11.27

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

8

2026.01.30

c++ 字符串格式化
c++ 字符串格式化

本专题整合了c++字符串格式化用法、输出技巧、实践等等内容,阅读专题下面的文章了解更多详细内容。

9

2026.01.30

java 字符串格式化
java 字符串格式化

本专题整合了java如何进行字符串格式化相关教程、使用解析、方法详解等等内容。阅读专题下面的文章了解更多详细教程。

8

2026.01.30

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 3.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

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

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