什么是元类呢?一切源自于一句话:python中一切皆为对象。让我们先定义一个类,然后逐步分析
| class StanfordTeacher(object): |
| school='Stanford' |
| |
| def __init__(self,name,age): |
| self.name=name |
| self.age=age |
| |
| def say(self): |
| print('%s says welcome to the Stanford to learn Python' %self.name) |
所有的对象都是实例化或者说调用类而得到的(调用类的过程称为类的实例化),比如对象t1是调用类StanfordTeacher得到的
| t1=StanfordTeacher('lili',18) |
| print(type(t1)) |
如果一切皆为对象,那么类StanfordTeacher本质也是一个对象,既然所有的对象都是调用类得到的,那么StanfordTeacher必然也是调用了一个类得到的,这个类称为元类
于是我们可以推导出===>产生StanfordTeacher的过程一定发生了:StanfordTeacher=元类(…)
| print(type(StanfordTeacher)) |
上文我们基于python中一切皆为对象的概念分析出:我们用class关键字定义的类本身也是一个对象,负责产生该对象的类称之为元类(元类可以简称为类的类),内置的元类为type
class关键字在帮我们创建类时,必然帮我们调用了元类StanfordTeacher=type(…),那调用type时传入的参数是什么呢?必然是类的关键组成部分,一个类有三大组成部分,分别是
1、类名class_name=’StanfordTeacher’
2、基类们class_bases=(object,)
3、类的名称空间class_dic,类的名称空间是执行类体代码而得到的
调用type时会依次传入以上三个参数
综上,class关键字帮我们创建一个类应该细分为以下四个过程
插图:元类原理之类的创建流程
补充:exec的用法
| |
| |
| |
| |
| |
| |
| |
| |
| |
| g={ |
| 'x':1, |
| 'y':2 |
| } |
| l={} |
| |
| exec(''' |
| global x,z |
| x=100 |
| z=200 |
| |
| m=300 |
| ''',g,l) |
| |
| print(g) |
| print(l) |
一个类没有声明自己的元类,默认他的元类就是type,除了使用内置元类type,我们也可以通过继承type来自定义元类,然后使用metaclass关键字参数为一个类指定元类
| class Mymeta(type): |
| pass |
| |
| |
| class StanfordTeacher(object,metaclass=Mymeta): |
| school='Stanford' |
| |
| def __init__(self,name,age): |
| self.name=name |
| self.age=age |
| |
| def say(self): |
| print('%s says welcome to the Stanford to learn Python' %self.name) |
自定义元类可以控制类的产生过程,类的产生过程其实就是元类的调用过程,即StanfordTeacher=Mymeta(‘StanfordTeacher’,(object),{…}),调用Mymeta会先产生一个空对象StanfordTeacher,然后连同调用Mymeta括号内的参数一同传给Mymeta下的__init__方法,完成初始化,于是我们可以
| class Mymeta(type): |
| def __init__(self,class_name,class_bases,class_dic): |
| |
| |
| |
| super(Mymeta, self).__init__(class_name, class_bases, class_dic) |
| |
| if class_name.islower(): |
| raise TypeError('类名%s请修改为驼峰体' %class_name) |
| |
| if '__doc__' not in class_dic or len(class_dic['__doc__'].strip(' \n')) == 0: |
| raise TypeError('类中必须有文档注释,并且文档注释不能为空') |
| |
| |
| class StanfordTeacher(object,metaclass=Mymeta): |
| """ |
| 类StanfordTeacher的文档注释 |
| """ |
| school='Stanford' |
| |
| def __init__(self,name,age): |
| self.name=name |
| self.age=age |
| |
| def say(self): |
| print('%s says welcome to the Stanford to learn Python' %self.name) |
储备知识:__call__
| class Foo: |
| def __call__(self, *args, **kwargs): |
| print(self) |
| print(args) |
| print(kwargs) |
| |
| obj=Foo() |
| |
| |
| res=obj(1,2,3,x=1,y=2) |
由上例得知,调用一个对象,就是触发对象所在类中的__call__方法的执行,如果把StanfordTeacher也当做一个对象,那么在StanfordTeacher这个对象的类中也必然存在一个__call__方法
| class Mymeta(type): |
| def __call__(self, *args, **kwargs): |
| print(self) |
| print(args) |
| print(kwargs) |
| return 123 |
| |
| class StanfordTeacher(object,metaclass=Mymeta): |
| school='Stanford' |
| |
| def __init__(self,name,age): |
| self.name=name |
| self.age=age |
| |
| def say(self): |
| print('%s says welcome to the Stanford to learn Python' %self.name) |
| |
| |
| |
| |
| t1=StanfordTeacher('lili',18) |
| print(t1) |
默认地,调用t1=StanfordTeacher(‘lili’,18)会做三件事
1、产生一个空对象obj
2、调用__init__方法初始化对象obj
3、返回初始化好的obj
对应着,StanfordTeacher类中的__call__方法也应该做这三件事
| class Mymeta(type): |
| def __call__(self, *args, **kwargs): |
| |
| obj=self.__new__(self) |
| |
| |
| self.__init__(obj,*args,**kwargs) |
| |
| |
| return obj |
| |
| class StanfordTeacher(object,metaclass=Mymeta): |
| school='Stanford' |
| |
| def __init__(self,name,age): |
| self.name=name |
| self.age=age |
| |
| def say(self): |
| print('%s says welcome to the Stanford to learn Python' %self.name) |
| |
| t1=StanfordTeacher('lili',18) |
| print(t1.__dict__) |
上例的__call__相当于一个模板,我们可以在该基础上改写__call__的逻辑从而控制调用StanfordTeacher的过程,比如将StanfordTeacher的对象的所有属性都变成私有的
| class Mymeta(type): |
| def __call__(self, *args, **kwargs): |
| |
| obj=self.__new__(self) |
| |
| |
| self.__init__(obj,*args,**kwargs) |
| |
| |
| obj.__dict__={'_%s__%s' %(self.__name__,k):v for k,v in obj.__dict__.items()} |
| |
| return obj |
| |
| class StanfordTeacher(object,metaclass=Mymeta): |
| school='Stanford' |
| |
| def __init__(self,name,age): |
| self.name=name |
| self.age=age |
| |
| def say(self): |
| print('%s says welcome to the Stanford to learn Python' %self.name) |
| |
| t1=StanfordTeacher('lili',18) |
| print(t1.__dict__) |
上例中涉及到查找属性的问题,比如self.__new__,请看下一小节