当前位置: 首页 > news >正文

Python __new__()方法详解

__new__() 是一种负责创建类实例的静态方法,它无需使用 staticmethod 装饰器修饰,且该方法会优先 __init__() 初始化方法被调用。

一般情况下,覆写 __new__() 的实现将会使用合适的参数调用其超类的 super().__new__(),并在返回之前修改实例。例如:

 
  1. class demoClass:
  2. instances_created = 0
  3. def __new__(cls,*args,**kwargs):
  4. print("__new__():",cls,args,kwargs)
  5. instance = super().__new__(cls)
  6. instance.number = cls.instances_created
  7. cls.instances_created += 1
  8. return instance
  9. def __init__(self,attribute):
  10. print("__init__():",self,attribute)
  11. self.attribute = attribute
  12. test1 = demoClass("abc")
  13. test2 = demoClass("xyz")
  14. print(test1.number,test1.instances_created)
  15. print(test2.number,test2.instances_created)

输出结果为:

__new__(): <class '__main__.demoClass'> ('abc',) {}
__init__(): <__main__.demoClass object at 0x0000026FC0DF8080> abc
__new__(): <class '__main__.demoClass'> ('xyz',) {}
__init__(): <__main__.demoClass object at 0x0000026FC0DED358> xyz
0 2
1 2


__new__() 通常会返回该类的一个实例,但有时也可能会返回其他类的实例,如果发生了这种情况,则会跳过对 __init__() 方法的调用。而在某些情况下(比如需要修改不可变类实例(Python 的某些内置类型)的创建行为),利用这一点会事半功倍。比如:

 
  1. class nonZero(int):
  2. def __new__(cls,value):
  3. return super().__new__(cls,value) if value != 0 else None
  4. def __init__(self,skipped_value):
  5. #此例中会跳过此方法
  6. print("__init__()")
  7. super().__init__()
  8. print(type(nonZero(-12)))
  9. print(type(nonZero(0)))

运行结果为:

__init__()
<class '__main__.nonZero'>
<class 'NoneType'>


那么,什么情况下使用 __new__() 呢?答案很简单,在 __init__() 不够用的时候。

例如,前面例子中对 Python 不可变的内置类型(如 int、str、float 等)进行了子类化,这是因为一旦创建了这样不可变的对象实例,就无法在 __init__() 方法中对其进行修改。

有些读者可能会认为,__new__() 对执行重要的对象初始化很有用,如果用户忘记使用 super(),可能会漏掉这一初始化。虽然这听上去很合理,但有一个主要的缺点,即如果使用这样的方法,那么即便初始化过程已经是预期的行为,程序员明确跳过初始化步骤也会变得更加困难。不仅如此,它还破坏了“__init__() 中执行所有初始化工作”的潜规则。

注意,由于 __new__() 不限于返回同一个类的实例,所以很容易被滥用,不负责任地使用这种方法可能会对代码有害,所以要谨慎使用。一般来说,对于特定问题,最好搜索其他可用的解决方案,最好不要影响对象的创建过程,使其违背程序员的预期。比如说,前面提到的覆写不可变类型初始化的例子,完全可以用工厂​方法(一种设计模式)来替代​。

Python中大量使用 __new__() 方法且合理的,就是 MetaClass 元类。有关元类的介绍

http://www.lryc.cn/news/209589.html

相关文章:

  • 虹科 | 解决方案 | 汽车示波器 索赔管理方案
  • 详解Jmeter中的BeanShell脚本
  • 前端和后端 优化
  • C++编译与运行:其二、编译期和运行期的区别
  • 汽车电子专有名词与相应技术
  • idea 没加载 provided的包
  • Hover:借贷新势力崛起,在经验与创新中找寻平衡
  • 软件设计原则-依赖倒置原则讲解以及代码示例
  • Linux--进程替换
  • 【计算机网络】TCP协议
  • 机器学习数据集:Kaggle
  • 软考 系统架构设计师系列知识点之设计模式(4)
  • PyCharm 安装 cx_Oracle 失败
  • 解决Windows出现找不到mfcm90u.dll无法打开软件程序的方法
  • 如何设计线程安全的 HashMap?
  • rpc汇总
  • OpenCV学习(五)——图像基本操作(访问图像像素值、图像属性、感兴趣区域ROI和图像边框)
  • 指针仪表读数YOLOV8NANO
  • 10000字!图解机器学习特征工程
  • Java 官方提供了哪几种线程池,分别有什么特点?
  • DTI-ALPS处理笔记
  • LVS集群-NAT模式
  • 微服务技术导学
  • p5.js 开发点彩画派的绘画工具
  • Java工具库——Commons IO的50个常用方法
  • Git: 仓库clone和用户配置
  • 构建外卖小程序:技术要点和实际代码
  • ubuntu安装配置svn
  • 『Jmeter入门万字长文』 | 从环境搭建、脚本设计、执行步骤到生成监控报告完整过程
  • Unity C#中LuaTable、LuaArrayTable、LuaDictTable中数据的增删改查