python

关注公众号 jb51net

关闭
首页 > 脚本专栏 > python > Python  __init__、__new__ 和 __call__

Python中的 __init__、__new__ 和 __call__示例详解

作者:哈拉索zbc

这篇文章主要介绍了Python中__init__、__new__ 和 __call__的相关资料,它们分别负责对象创建、初始化和调用,用于实现单例模式、装饰器等高级功能,需要的朋友可以参考下

这三个特殊方法是 Python 面向对象编程中的核心构造机制,分别在对象生命周期的不同阶段发挥作用:

1.__init__方法:对象初始化器

2.__new__方法:对象创建器

3.__call__方法:对象调用器

class Adder:
    def __init__(self, base):
        self.base = base
    
    def __call__(self, x):
        print("__call__ invoked")
        return self.base + x

# 使用
add5 = Adder(5)  # 创建可调用对象
print(add5(3))   # 输出: __call__ invoked -> 8
print(add5(10))  # 输出: __call__ invoked -> 15

高级应用:装饰器类

class TraceCalls:
    def __init__(self, func):
        self.func = func
        self.call_count = 0
    
    def __call__(self, *args, **kwargs):
        self.call_count += 1
        print(f"Call #{self.call_count} to {self.func.__name__}")
        return self.func(*args, **kwargs)

@TraceCalls
def multiply(a, b):
    return a * b

print(multiply(3, 4))
# 输出: 
# Call #1 to multiply
# 12

print(multiply(5, 6))
# 输出:
# Call #2 to multiply
# 30

三者的协作流程

class Example:
    def __new__(cls, *args, **kwargs):
        print("1. __new__: Creating instance")
        instance = super().__new__(cls)
        return instance
    
    def __init__(self, value):
        print("2. __init__: Initializing instance")
        self.value = value
    
    def __call__(self, multiplier):
        print("3. __call__: Invoking instance")
        return self.value * multiplier

# 完整生命周期
e = Example(10)  
# 输出:
# 1. __new__: Creating instance
# 2. __init__: Initializing instance

result = e(5)    
# 输出:
# 3. __call__: Invoking instance
print(result)  # 50

关键对比表

特性__new____init____call__
角色对象创建器对象初始化器对象调用器
调用时机创建实例时(最先)__new__ 之后实例被调用时
首参数cls (类引用)self (实例引用)self (实例引用)
返回值必须返回实例对象必须返回 None可返回任意值
主要用途控制实例创建过程初始化实例属性使实例可像函数一样调用
是否必需不定义则用 object.__new__不定义则跳过初始化不定义则实例不可调用
典型场景单例模式、不可变类型常规对象初始化装饰器类、函数对象

高级应用场景

1. 不可变类型扩展

class UppercaseStr(str):
    def __new__(cls, value):
        # 在创建前修改值
        return super().__new__(cls, value.upper())

s = UppercaseStr("hello")
print(s)  # "HELLO"(不可变对象在创建时确定值)

2. 对象工厂

class Animal:
    def __new__(cls, animal_type, *args):
        if animal_type == "dog":
            return Dog(*args)
        elif animal_type == "cat":
            return Cat(*args)
        raise ValueError(f"Unknown animal: {animal_type}")

class Dog:
    def __init__(self, name):
        self.name = name

class Cat:
    def __init__(self, name):
        self.name = name

d = Animal("dog", "Buddy")
print(type(d), d.name)  # <class '__main__.Dog'> Buddy

3. 带状态的装饰器

class Retry:
    def __init__(self, max_attempts=3):
        self.max_attempts = max_attempts
    
    def __call__(self, func):
        def wrapper(*args, **kwargs):
            for attempt in range(1, self.max_attempts+1):
                try:
                    return func(*args, **kwargs)
                except Exception as e:
                    print(f"Attempt {attempt} failed: {e}")
            raise RuntimeError("All attempts failed")
        return wrapper

@Retry(max_attempts=2)
def risky_operation():
    import random
    if random.random() < 0.7:
        raise ValueError("Random failure")
    return "Success"

print(risky_operation())

常见陷阱与解决方案

1. __init__ 中忘记返回 None

class Mistake:
    def __init__(self):
        return 42  # TypeError: __init__() should return None

2.__new__ 忘记返回实例

class BadClass:
    def __new__(cls):
        print("Forgot to return instance")  # 返回None -> TypeError

3.单例模式中的重复初始化

解决方案:使用标志位控制

class SafeSingleton:
    _instance = None
    _initialized = False
    
    def __new__(cls):
        if not cls._instance:
            cls._instance = super().__new__(cls)
        return cls._instance
    
    def __init__(self):
        if not self.__class__._initialized:
            # 初始化代码
            self.__class__._initialized = True

4.__call__ 与 __init__ 参数混淆

掌握这三个特殊方法的区别和协作机制,能够让你更精细地控制 Python 对象的整个生命周期,实现更高级的设计模式。

总结

到此这篇关于Python中的 __init__、__new__ 和 __call__的文章就介绍到这了,更多相关Python __init__、__new__ 和 __call__内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文