【python进阶】详解元类及其应用2
前言
在上一篇文章【python进阶】详解元类及其应用1中,我们提到了关于元类的一些前置知识,介绍了类对象,动态创建类,使用type创建类,这一节我们将继续接着上文来讲~~~
5.使⽤type创建带有⽅法的类
最终你会希望为你的类增加⽅法。只需要定义⼀个有着恰当签名的函数并将 其作为属性赋值就可以了。
添加实例⽅法
In [14]: def echo_bar(self):#定义了一个普通的函数 ...: print(self.bar) ...: In [15]: FooChild = type('FooChild',(Foo,),{'echo_bar':echo_bar}) In [16]: hasattr(Foo,'echo_bar')#判断Foo类中,是否有echo_bar这个属性 Out[16]: False In [17]: hasattr(FooChild,'echo_bar')#判断FooChild类中,是否有echo_bar这个属性 Out[17]: True In [18]: my_foo=FooChild() ...: In [19]: my_foo.echo_bar() True
添加静态⽅法
In [20]: @staticmethod ...: def testStatic(): ...: print("static method ....") ...: In [21]: Foochild=type('Foochild',(Foo,),{"echo_bar":echo_bar,"testStatic":testStatic}) In [22]: fooclid=Foochild() In [23]: fooclid.testStatic Out[23]: <function __main__.testStatic> In [24]: fooclid.testStatic() static method .... In [25]: fooclid.echo_bar() True
添加类⽅法
In [26]: @classmethod ...: def testClass(cls): ...: print(cls.bar) ...: In [27]: Foochild=type('Foochild',(Foo,),{"echo_bar":echo_bar,"testStatic":testStatic,"testClass":testClass}) In [28]: fooclid = Foochild() In [29]: fooclid.testClass() True
你可以看到,在Python中,类也是对象,你可以动态的创建类。这就是当你 使⽤关键字class时Python在幕后做的事情,⽽这就是通过元类来实现的。
6.到底什么是元类(终于到主题了)
元类就是⽤来创建类的“东⻄”。你创建类就是为了创建类的实例对象,不是吗?但是我们已经学习到了Python中的类也是对象。
元类就是⽤来创建这些类(对象)的,元类就是类的类,你可以这样理解为:
MyClass = MetaClass()#使⽤元类创建出⼀个对象,这个对象称为“类” MyObject = MyClass()#使⽤“类”来创建出实例对象
你已经看到了type可以让你像这样做:
MyClass=type('MyClass',(),{})
这是因为函数type实际上是⼀个元类。type就是Python在背后⽤来创建所有类的元类。现在你想知道那为什么type会全部采⽤⼩写形式⽽不是Type呢? 好吧,我猜这是为了和str保持⼀致性,str是⽤来创建字符串对象的类,⽽int 是⽤来创建整数对象的类。type就是创建类对象的类。你可以通过检查 __class__属性来看到这⼀点。Python中所有的东⻄,注意,我是指所有的东 ⻄——都是对象。这包括整数、字符串、函数以及类。它们全部都是对象, ⽽且它们都是从⼀个类创建⽽来,这个类就是type。
In [30]: age = 35 In [31]: age.__class__ Out[31]: int In [32]: name = 'bob' In [33]: name.__class__ Out[33]: str In [34]: def foo(): ...: pass ...: In [35]: foo.__class__ Out[35]: function In [36]: class Bar(object): ...: pass ...: In [37]: b = Bar() In [38]: b.__class__ Out[38]: __main__.Bar
现在,对于任何⼀个__class__的__class__属性⼜是什么呢?
In [40]: name.__class__.__class__ Out[40]: type In [41]: age.__class__.__class__ Out[41]: type In [42]: foo.__class__.__class__ Out[42]: type In [43]: b.__class__.__class__ Out[43]: type
因此,元类就是创建类这种对象的东⻄。type就是Python的内建元类,当然 了,你也可以创建⾃⼰的元类。
7.__metaclass__属性
你可以在定义⼀个类的时候为其添加__metaclass__属性。
class Foo(object): __metaclass__ = something... ...省略...
如果你这么做了,Python就会⽤元类来创建类Foo。⼩⼼点,这⾥⾯有些技 巧。你⾸先写下class Foo(object),但是类Foo还没有在内存中创建。Python 会在类的定义中寻找__metaclass__属性,如果找到了,Python就会⽤它来创建类Foo,如果没有找到,就会⽤内建的type来创建这个类。把下⾯这段话反复读⼏次。当你写如下代码时 :
In [44]: class Foo(Bar): ...: pass ...:
Python做了如下的操作:
- Foo中有__metaclass__这个属性吗?如果是,Python会通过 __metaclass__创建⼀个名字为Foo的类(对象)
- 如果Python没有找到__metaclass__,它会继续在Bar(⽗类)中寻找 __metaclass__属性,并尝试做和前⾯同样的操作。
-
如果Python在任何⽗类中都找不到__metaclass__,它就会在模块层次 中去寻找__metaclass__,并尝试做同样的操作。
-
如果还是找不到__metaclass__,Python就会⽤内置的type来创建这个类对象。
现在的问题就是,你可以在__metaclass__中放置些什么代码呢?答案就 是:可以创建⼀个类的东⻄。那么什么可以⽤来创建⼀个类呢?type,或者任何使⽤到type或者⼦类化type的东东都可以。
8.⾃定义元类
元类的主要⽬的就是为了当创建类时能够⾃动地改变类。通常,你会为API做 这样的事情,你希望可以创建符合当前上下⽂的类。
假想⼀个很傻的例⼦,你决定在你的模块⾥所有的类的属性都应该是⼤写形 式。有好⼏种⽅法可以办到,但其中⼀种就是通过在模块级别设定 __metaclass__。采⽤这种⽅法,这个模块中的所有类都会通过这个元类来创建,我们只需要告诉元类把所有的属性都改成⼤写形式就万事⼤吉了。
幸运的是,__metaclass__实际上可以被任意调⽤,它并不需要是⼀个正式的类。所以,我们这⾥就先以⼀个简单的函数作为例⼦开始。
python3中
#-*-coding:utf-8-*- def upper_attr(future_class_name,future_class_parents,future_class_attr): #遍历属性字典,把不是__开头的属性名字变为⼤写 newAttr = {} for name,value in future_class_attr.items(): if not name.startswith("__"): newAttr[name.upper()] = value #调⽤type来创建⼀个类 return type(future_class_name,future_class_parents,newAttr) class Foo(object,metaclass=upper_attr): bar = 'bip' print(hasattr(Foo,'bar')) print(hasattr(Foo,'BAR')) f = Foo() print(f.BAR)
现在让我们再做⼀次,这⼀次⽤⼀个真正的class来当做元类。
#-*-coding:utf-8-*- class UpperAttrMetaClass(type): #__new__是在__init__之前被调⽤的特殊⽅法 #__new__是⽤来创建对象并返回之的⽅法 #⽽__init__只是⽤来将传⼊的参数初始化给对象 #你很少⽤到__new__,除⾮你希望能够控制对象的创建 #这⾥,创建的对象是类,我们希望能够⾃定义它,所以我们这⾥改写__new__ #如果你希望的话,你也可以在__init__中做些事情 #还有⼀些⾼级的⽤法会涉及到改写__call__特殊⽅法,但是我们这⾥不⽤ def __new__(cls,future_class_name,future_class_parents,future_class_attr): #遍历属性字典,把不是__开头的属性名字变为⼤写 newAttr = {} for name,value in future_class_attr.items(): if not name.startswith("__"): newAttr[name.upper()] = value #⽅法1:通过'type'来做类对象的创建 #return type(future_class_name,future_class_parents,newAttr) #⽅法2:复⽤type.__new__⽅法 #这就是基本的OOP编程,没什么魔法 #return type.__new__(cls,future_class_name,future_class_parents,future_class_attr) #⽅法3:使⽤super⽅法 return super(UpperAttrMetaClass,cls).__new__(cls,future_class_name,future_class_parents,future_class_attr) #python2的⽤法 #class Foo(object): # __metaclass__ = upper_attr#设置Foo类的元类为upper_attr # bar = 'bip' #python3的⽤法 class Foo(object,metaclass=upper_attr): bar = 'bip' print(hasattr(Foo,'bar')) #输出:False print(hasattr(Foo,'BAR')) #输出:True f = Foo() print(f.BAR) #输出:'bip'
就是这样,除此之外,关于元类真的没有别的可说的了。但就元类本身⽽ ⾔,它们其实是很简单的:
- 拦截类的创建
- 修改类
- 返回修改之后的类
究竟为什么要使⽤元类?
现在回到我们的⼤主题上来,究竟是为什么你会去使⽤这样⼀种容易出错且晦涩的特性?好吧,⼀般来说,你根本就⽤不上它:
“元类就是深度的魔法,99%的⽤户应该根本不必为此操⼼。如果你想搞清楚 究竟是否需要⽤到元类,那么你就不需要它。那些实际⽤到元类的⼈都⾮常 清楚地知道他们需要做什么,⽽且根本不需要解释为什么要⽤元类。” —— Python界的领袖 Tim Peters
您可以考虑给博主来个小小的打赏以资鼓励,您的肯定将是我最大的动力。thx.
微信打赏
支付宝打赏
作 者: Angel_Kitty
出 处:http://www.cnblogs.com/ECJTUACM-873284962/
关于作者:潜心机器学习以及信息安全的综合研究。如有问题或建议,请多多赐教!
版权声明:本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。
特此声明:所有评论和私信都会在第一时间回复。也欢迎园子的大大们指正错误,共同进步。或者直接私信我
声援博主:如果您觉得文章对您有帮助,可以点击右下角【推荐】推荐一下该博文。您的鼓励是作者坚持原创和持续写作的最大动力!

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Java对时间进行操作(LocalDateTime,Calendar)
题记: Java的时间日期API一直以来都是被诟病的东西,为了解决这一问题,Java 8中引入了新的时间日期API,其中包括LocalDate、LocalTime、LocalDateTime、Clock、Instant等类,这些的类的设计都使用了不变模式,因此是线程安全的设计。 一、如下有详细的说明: package com.gws; import java.text.SimpleDateFormat; import java.time.Clock; import java.time.LocalDate; import java.time.LocalDateTime; import java.time.format.DateTimeFormatter; import java.util.Calendar; import java.util.Date; import java.util.TimeZone; /** * @author:wangdong * @description:Java对日期和时间的处理 */ public class DateTimeTest { public st...
- 下一篇
读书节最该买的书,我都帮你们挑出来了
点击关注异步图书,置顶公众号 每天与你分享 IT好书 技术干货 职场知识 过完漫长的冬天,送走了倒春寒,转眼4月也即将过半,我们有那么多的节日要过,对爱读书的真爱粉儿而言,读书节这个大日子,不放点福利哪行? 就在昨天异步社区掌柜的发布异步社区3.X版本,社区做了近三年最大的一次升级改版, 还没看过的读者请看这篇《异步社区进入3.X时代!这些新玩法你一定要知道》,其中不仅仅优化了UI界面、写作环境,最最关键的重点就是:异步社区实现了多介质内容服务,不仅可以购买纸质书还可以购买电子书(e读版电子书和推送版电子书)、视频课程;同时不管你在哪里购买了异步图书,只要扫描印在图书上的二维码,将图书加入“我的书架”,就能够随时接收到这本书相关的信息推送和服务,还会有优惠购买本书电子书的专享权益。 话说到这儿,该说说福利啦!异步社区全场满99减20元/199减50元。同时来一组近期值得买的书单,如果你还没入手,这次机会不可错过。 编程新书 纸书/电子书同步 《Python神经网络编程》 [英]塔里克·拉希德(Tariq Rashid)著 点击封面购买纸书 当前,深度学习和人工智能的发展和应...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS8安装MyCat,轻松搞定数据库的读写分离、垂直分库、水平分库
- SpringBoot2整合Thymeleaf,官方推荐html解决方案
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS关闭SELinux安全模块
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- Eclipse初始化配置,告别卡顿、闪退、编译时间过长
- SpringBoot2整合Redis,开启缓存,提高访问速度
- CentOS7设置SWAP分区,小内存服务器的救世主