Python元类行为(不调用__new__),有解释吗?
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了Python元类行为(不调用__new__),有解释吗?,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含2562字,纯文字阅读大概需要4分钟。
内容图文
在一个名为exp.py(如下)的文件中,我试图了解Python中的元类.好像当元类的__new__方法使用’type’构造函数构造一个类时,使用它作为元类的类的子类不会调用其__new__方法:
class A(type):
def __new__(cls, name, bases, dct):
print "cls is: ", cls
print "name is: ", name
print "bases is: ", bases
print "dct is: ", dct
print
return super(A, cls).__new__(cls, name, bases, dct)
class B(object):
__metaclass__ = A
class C(B): pass
class D(type):
def __new__(cls, name, bases, dct):
print "cls is: ", cls
print "name is: ", name
print "bases is: ", bases
print "dct is: ", dct
return type(name, bases, dct)
class E(object):
__metaclass__ = D
class F(E): pass
在终端中:
>>> from exp import *
cls is: <class 'exp.A'>
name is: B
bases is: (<type 'object'>,)
dct is: {'count': 0, '__module__': 'exp', '__metaclass__': <class 'exp.A'>, '__init__': <function __init__ at 0x107eb9578>}
cls is: <class 'exp.A'>
name is: C
bases is: (<class 'exp.B'>,)
dct is: {'__module__': 'exp'}
cls is: <class 'exp.D'>
name is: E
bases is: (<type 'object'>,)
dct is: {'count': 0, '__module__': 'exp', '__metaclass__': <class 'exp.D'>, '__init__': <function __init__ at 0x107ebdb18>}
>>>
如您所见,加载类F的定义时,不会调用元类D的__new__方法. (如果已被调用,则也将打印有关F类的信息.)
有人可以帮我解释一下吗?
相关文章:我正在阅读What is a metaclass in Python?,并且在句子中似乎遇到了类似的内容,“请注意,这里不会继承__metaclass__属性,父级(Bar .__ class__)的元类将是继承的.如果Bar使用__metaclass__属性使用type()(而不是type .__ new __())创建Bar的子类将不会继承该行为.”但是我对此并不完全理解.
解决方法:
在第二种情况下,您实际上不是在返回类型的实例,而是返回了元类的实例,而是依次将新创建的类E的__class__属性设置为< type'type'>.因此as per the rule 2 Python会检查基类的__class__属性(或type(E)或E .__ class__),并决定使用type作为F的元类,因此在这种情况下永远不会调用D的__new__.
>如果dict [‘__ metaclass__’]存在,则使用它.
>否则,如果存在至少一个基类,则使用其元类(此方法首先查找__class__属性,如果找不到,则使用其类型).
因此,正确的方法是在元类的__new__方法中调用类型的__new__方法:
class D(type):
def __new__(cls, name, bases, dct):
print "cls is: ", cls
print "name is: ", name
print "bases is: ", bases
print "dct is: ", dct
return type.__new__(cls, name, bases, dct)
class E(object):
__metaclass__ = D
class F(E): pass
class A(object):
pass
输出:
cls is: <class '__main__.D'>
name is: E
bases is: (<type 'object'>,)
dct is: {'__module__': '__main__', '__metaclass__': <class '__main__.D'>}
cls is: <class '__main__.D'>
name is: F
bases is: (<class '__main__.E'>,)
dct is: {'__module__': '__main__'}
内容总结
以上是互联网集市为您收集整理的Python元类行为(不调用__new__),有解释吗?全部内容,希望文章能够帮你解决Python元类行为(不调用__new__),有解释吗?所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。