用于__new__的 Python 类
本篇主要想要详细的介绍一下关于类的魔法方法__new__()方法。
在学习之前我们看一下Python3中关于object基类的__new__() 方法:
@staticmethod # known case of __new__ def __new__(cls, *more): # known special case of object.__new__ """ Create and return a new object. See help(type) for accurate signature. """ pass
上述描述:__new__()创建和返回一个新的对象。通俗说:该魔法属性是用来创建实例对象的。接下来我们看一下它的是如何创建对象的。
一、理解
1、__new__()是在新式类中新出现的方法,它作用在构造方法建造实例之前。
即可以这样理解:Python中存在于类中的构造方法__init__()负责将类实例化,而在__init__()执行之前,__new__()负责制造这样的一个实例对象,以便__init__()去让该实例对象更加的丰富(为其添加属性等)。
同时:__new__() 方法还决定是否要使用该__init__() 方法,因为__new__()可以调用其他类的构造方法或者直接返回别的对象来作为本类 的实例。
2、__new__() 方法始终都是类的静态方法,即使没有被加上静态方法装饰器。
比喻
如果将类比喻为工厂,那么__init__()方法则是该工厂的生产工人,__init__()方法接受的初始化参 数则是生产所需原料,__init__()方法会按照方法中的语句负责将原料加工成实例以供工厂出货。而 __new__()则是生产部经理,__new__()方法可以决定是否将原料提供给该生产部工人,同时它还决定着出 货产品是否为该生产部的产品,因为这名经理可以借该工厂的名义向客户出售完全不是该工厂的产品。 |
二、结构
在Python中,若不重写类中的__new__()方法,其默认的结构时这样的:
class Foo(object): def __new__(cls,*args,**kwagrs): return super().__new__(cls,*args,**kwagrs)
1、其第一个参数 cls 是当前正在实例化的类。
class Foo(object): def __new__(cls,*args,**kwagrs): return object.__new__(cls,*args,**kwagrs)
注:①、如果(新式)类中没有重写__new__()方法,即在定义新式类时没有重新定义__new__()时 ,Python默认是调用该类的直接父类的__new__()方法来构造该类的实例,如果该类的父类也没有重写 __new__(),那么将一直按此规矩追溯至object的__new__()方法,因为object是所有新式类的基类。
②、如果新式类中重写了__new__()方法,那么你可以*选择任意一个的其他的新式类(必定要是 新式类,只有新式类必定都有__new__(),因为所有新式类都是object的后代,而经典类则没有__new__() 方法)的__new__()方法来制造实例,包括这个新式类的所有前代类和后代类,只要它们不会造成递归死 循环。
class A(object): pass class B(A): pass class C(B): def __new__(cls,*args,**kwargs): return super().__new__(cls, *args,**kwargs) # 等同于: # return B.__new__(cls, *args,**kwargs) # return A.__new__(cls, *args,**kwargs) # return object.__new__(cls, *args,**kwargs)
2、在任何新式类的__new__()方法,不能调用自身的__new__()来制造实例,因为这会造成死循环。
例如:
class Bar(object): def __new__(cls,*agrs,**kwagrs): return Bar.__new__(Foo,*agrs,**kwagrs)
这样的话会造成 当Bar调用其自身的__new__() 方法制造实例时,又会跳到第二行去执行__new__(),当到第三行时,又跳到第二行去调用__new__(),这样不断的就会陷入死循环。
3、Bar.__new__(Foo,*agrs,**kwagrs) ,指:使用哪个类(bar)的__new__()方法去制造谁的(Foo)实例对象。
通常来说,新式类开始实例化时,__new__()方法会返回cls(cls指代当前类)的实例,然后该类的 __init__()方法作为构造方法会接收这个实例(即self)作为自己的第一个参数,然后依次传入__new__ ()方法中接收的位置参数和命名参数。
class Foo(object): def __new__(cls,*args,**kwagrs): return object.__new__(cls,*args,**kwagrs) def __init__(self,name): self.name = name class Bar(object): def __new__(cls,*agrs,**kwagrs): return object.__new__(Foo,*agrs,**kwagrs) bar = Bar() print(type(bar)) #foo其实是Stranger类的实例。 # 输出为:<class '__main__.Foo'>
会发现上例:制造的是Foo的实例对象。
总结:因此可以这么描述__new__()和__ini__()的区别,在新式类中__new__()才是真正的实例化方法,为类提供外壳制造出实例框架,然后调用该框架内的构造方法__init__()使其丰满。
上一篇: Python 中的类和类型
下一篇: Python 类属性使用汇总
推荐阅读
-
4.4 章 python 中的循环结构必须相互嵌套:常用于形状(矩形、三角形、菱形)。
-
Java 8新特性探究(十三)JavaFX 8新特性以及开发2048游戏-JavaFX历史## 跟java在服务器端和web端成绩相比,桌面一直是java的软肋,于是Sun公司在2008年推出JavaFX,弥补桌面软件的缺陷,请看下图JavaFX一路走过来的改进 从上图看出,一开始推出时候,开发者需使用一种名为JavaFX Script的静态的、声明式的编程语言来开发JavaFX应用程序。因为JavaFX Script将会被编译为Java bytecode,程序员可以使用Java代码代替。 JavaFX 2.0之后的版本摒弃了JavaFX Script语言,而作为一个Java API来使用。因此使用JavaFX平台实现的应用程序将直接通过标准Java代码来实现。 JavaFX 2.0 包含非常丰富的 UI 控件、图形和多媒体特性用于简化可视化应用的开发,WebView可直接在应用中嵌入网页;另外 2.0 版本允许使用 FXML 进行 UI 定义,这是一个脚本化基于 XML 的标识语言。 从JDK 7u6开始,JavaFx就与JDK捆绑在一起了,JavaFX团队称,下一个版本将是8.0,目前所有的工作都已经围绕8.0库进行。这是因为JavaFX将捆绑在Java 8中,因此该团队决定跳过几个版本号,迎头赶上Java 8。 ##JavaFx8的新特性 ## ###全新现代主题:Modena 新的Modena主题来替换原来的Caspian主题。不过在Application的start方法中,可以通过setUserAgentStylesheet(STYLESHEET_CASPIAN)来继续使用Caspian主题。 参考http://fxexperience.com/2013/03/modena-theme-update/ ###JavaFX 3D 在JavaFX8中提供了3D图像处理API,包括Shape3D (Box, Cylinder, MeshView, Sphere子类),SubScene, Material, PickResult, LightBase (AmbientLight 和PointLight子类),SceneAntialiasing等。Camera类也得到了更新。从JavaDoc中可以找到更多信息。 ###富文本 强化了富文本的支持 ###TreeTableView ###日期控件DatePicker 增加日期控件 ###用于 CSS 结构的公共 API
-
解析Python中的类的方法和概念
-
QRubberBand:适用于橡皮筋类功能的工具
-
逆滤波与维纳滤波的应用于Python运动模糊和退化模型的点扩散函数
-
在python机器学习中,classification_report函数用于输出模型评估报告,其中查全率的定义公式被包含
-
Python实战技巧分享:类与对象的常用写法
-
详解在React中运用react-router-dom的不同方法传递路由参数(适用于v5.x和v6.x版本) - 介绍三类路由值传输方式 (v6.x 版本)
-
如何在实践中综合运用TypeScript:创建一个能适用于MySQL、MongoDB和 MSSQL的底层类库,通过类型、接口、类及泛型实现统一操作
-
用Python实现的灰色聚类评价模型详细教程