【原创】Python 对象创建过程中元类, __new__, __call__, __init__ 的处理

原始type:

type是最原始的元类,其__call__方法是在你使用" t_class = type(classname_string, base_classes_tuple, attributes_dict)" 这种语法来使用时, 在__call__方法内使用又会调用type的__new__和__init__方法来创建classname_string的具体类,并初始化类信息。当type(***)调用完成, classname_string代表的类可以用来创建实例了。

元类调用过程: 原始type元类同理

如下流程:假设是MyMeta元类,而不是原始type元类

例子: MyClass = MyMeta(‘MyClass‘, bases, attributes)

my_meta_type = type(MyMeta)MyClass= my_meta_type.__call__(MyMeta, cls, bases, attributes)在__call__中应该是如下操作:MyClass = MyMeta.__new__(MyMeta, cls, bases, attributes)
meta_class = MyClass.__metaclass__
meta_class.__init__(MyClass, cls, bases, attributes)
return MyClass

最终返回MyClass类

上述元类有一个很令人迷惑的地方,需要注意到,当你的元类是自定义的元类的时候,假设是MyMeta,此时调用的是MyMeta的父元类type的__call__,所以假设MyMeta自定义了__call__,你要知道当调用MyMeta()的时候,该函数并没有被调用,调用的是type的__call__,你定义MyClass对象实例时才会调用该函数。

总结: 元类处理过程:定义一个类时,使用声明或者默认的元类对该类进行创建,对元类求type运算,得到父元类(该类声明的元类的父元类),调用父元类的__call__函数,在父元类的__call__函数中, 调用该类声明的元类的__new__函数来创建对象(该函数需要返回一个对象(指类)实例),然后再调用该元类的__init__初始化该对象(此处对象是指类,因为是元类创建的对象),最终返回该类。

你可以简单实验以下,自定义俩个元类,该俩个元类是父子关系,在定义一个类,设置使用自定义元类的子元类,发现会调用自定义元类的父元类中call的输出,子元类的call并没有输出,在定义类的对象时才输出了

例子如下:

class SuperMeta(type):

def __call__(metaname, clsname, baseclasses, attrs):

print ‘SuperMeta Called‘

clsob = type.__new__(metaname, clsname, baseclasses, attrs)

type.__init__(clsob, clsname, baseclasses, attrs)

return clsob

class MyMeta(type):

__metaclass__ = SuperMeta

def __call__(cls, *args, **kwargs):

print ‘MyMeta called‘, cls, args, kwargs

ob = object.__new__(cls, *args)

ob.__init__(*args)

return ob

print ‘create class‘

class Kls(object):

__metaclass__ = MyMeta

def __init__(self, data):

self.data = data

def printd(self):

print self.data

print ‘class created ---------------------‘

# 你会发现定义了 Kls 类后输出了 SuperMeta 父元类的输出

ik = Kls(‘arun‘)

ik.printd()

ik2 = Kls(‘avni‘)

ik2.printd()

# 定义Kls对象实例才真的执行了MyMeta的call

为什么type会调用自己的呢,因为type的type还是type, 蛋疼一小会……

附加:

原始type的__call__应该是参数结构应该是:

  metaname, clsname, baseclasses, attrs

原始type的__new__

  metaname, clsname, baseclasses, attrs

原始type的__init__

  class_obj, clsname, baseclasses, attrs

元类的__new__和__init__影响的是创建类对象的行为,父元类的__call__控制对子元类的 __new__,__init__的调用,就是说控制类对象的创建和初始化。父元类的__new__和__init__由更上层的控制,

    一般来说,原始type是最初的父元类,其__new__和__init__是具有普遍意义的,即应该是分配内存、初始化相关信息等

元类__call__影响的是创建类的实例对象的行为,此时如果类自定义了__new__和__init__就可以控制类的对象实例的创建和初始化

参考:

http://pythoncentral.io/how-metaclasses-work-technically-in-python-2-and-3/

http://stackoverflow.com/questions/2608708/what-is-the-difference-between-type-and-type-new-in-python  Florentin的答案

classKls(object):

__metaclass__=MyMeta

def__init__(self,data):

self.data=data

defprintd(self):

printself.data

时间: 2024-11-02 04:47:09

【原创】Python 对象创建过程中元类, __new__, __call__, __init__ 的处理的相关文章

Python中元类

一. 前提: python中一切都是对象,要么是类的对象,要么是元类的对象,type元类是自己的对象.继承层次表示为type(元类)-->类(内置和用户自定义的) --> 类的实例. 二.  目的: 创建类时自动改变类,换句话说就是创建类这种东西的东西. 三.  __metaclass__ 可以通过指定__metaclass__来使用用户自定义的元类创建类. class语句解析顺序: class Myclass(object): pass Myclass里面的__metaclass__ --&

对python中元类的理解

1. 类也是对象 在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段.在Python中这一点仍然成立: >>> class ObjectCreator(object): - pass - >>> my_object = ObjectCreator() >>> print my_object <__main__.ObjectCreator object at 0x8974f2c> 但是,Python中的类还远不止如此.类同样也是一

Java中对象创建过程

本文介绍的对象创建过程仅限于普通Java对象,不包括数组和Class对象. 1.类加载检查 虚拟机遇到一条new指令时,首先去检查该指令的参数能否在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载.解析和初始化过. 2.为新生对象分配内存 对象所需内存的大小在类加载完成后便可以完全确定,为对象分配空间的任务等同于把一块确定大小的内存从Java堆中划分出来.划分的方法主要有两种: a.指针碰撞 如果java堆中内存是绝对规整的,所有用过的内存都放在一边,未使用的内存放在另一

Python第二十一课(反射/元类)

Python第二十一课(反射/元类)    >>>思维导图>>>中二青年 反射reflect 什么是反射, 其实是反省,自省的意思 反射指的是一个对象应该具备,可以检测,修改,增加自身属性的能力 反射就是通过字符串操作属性 涉及的四个函数,这四个函数就是普通的内置函数 没有双下划綫,与print等等没有区别 hasattr getattr setattr delattr p = Person("jack",18,"man") #

【python进阶】详解元类及其应用2

前言 在上一篇文章[python进阶]详解元类及其应用1中,我们提到了关于元类的一些前置知识,介绍了类对象,动态创建类,使用type创建类,这一节我们将继续接着上文来讲~~~ 5.使?type创建带有?法的类 最终你会希望为你的类增加?法.只需要定义?个有着恰当签名的函数并将 其作为属性赋值就可以了.添加实例?法 In [14]: def echo_bar(self):#定义了一个普通的函数 ...: print(self.bar) ...: In [15]: FooChild = type('

VC++编程之第三课笔记——MFC窗口创建过程以及窗口类的封装

第三课 MFC窗口创建过程以及窗口类的封装 MFC的每一个类都是以C开头的,表明这是一个Class. 工程包含(单文档) 创建工程名为aaa的工程(单文档)时,在类视图中可看见五个类: CAboutDlg CMainFrame CAaaApp CAaaDoc CAaaView 其中: 类CAboutDlg继承自CDialog类,对话框的类 类CMainFrame继承自CFrameWnd类,创建整个程序的框架窗口 类CAaaApp继承自CWinApp类,创建唯一的应用程序对象 类CAaaDoc继承

虚拟机创建过程中镜像格式的的变化过程

这里通过OpenStack的horizon组件来创建一个m1.small的virtual machine,来详细分析下镜像格式的变化以及glance底层具体执行的哪些操作. (1)首先看一下Glance管理的镜像,如果采用local storage,glance将镜像文件默认存储到/var/lib/glance/image目录下,这里我们选择c036d689-0336-4fcd-a8e0-4aed4dd5e420这个镜像来作为创建虚拟机的模板,此镜像是通过如下命令添加的,因此在horizon中显

元类,__call__方法和单例模式

在python中一切皆对象的概念. 举一个例子: class Chinese: country="china" def __init__(self, name,age,sex): self.name=name self.age=age self.sex=sex def change(self): print('%s speak Chinese '%self.name) #一切皆对象的原则. 那么这个类Chinese也是对象,那么他也有自己的类.那么他的类就是type 元类:类的类就是元

《Java中对象创建过程》

http://jzinfo.iteye.com/blog/620045具体问题:java中使用new创建一个对象的时候,详细的过程是怎么样的. Java中每个Java代码要能执行首先会编译成一个class的字节码文件.然后利用类装载器装载进入JVM然后才能被执行. 1. 所有的类都是在对其第一次使用时,动态加载到JVM中.当首次创建类型为Dog的对象时,或者Dog类的静态方法,静态属性域首次被访问时,java解释器查找classPath,定位到Dog.class文件 2. 载入Dog.class