【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
【python进阶】详解元类及其应用2的更多相关文章
- Python进阶——详解元类,metaclass的原理和用法
本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是Python专题第18篇文章,我们来继续聊聊Python当中的元类. 在上上篇文章当中我们介绍了type元类的用法,在上一篇文章当中我 ...
- 【python进阶】详解元类及其应用1
前言 元类在python中是很重要的一部分,我将分两次去讲解元类及其应用,此篇为详解元类及其应用第一篇,下面开始今天的说明~~~ 1. 类也是对象 在⼤多数编程语⾔中,类就是⼀组⽤来描述如何⽣成⼀个对 ...
- Python说文解字_详解元类
1.深入理解一切接对象: 1.1 什么是类和对象? 首先明白元类之前要明白什么叫做类.类是面向对象object oriented programming的重要概念.在面向对象中类和对象是最基本的两个概 ...
- Python进阶开发之元类编程
系列文章 √第一章 元类编程,已完成 ; 本文目录 类是如何产生的如何使用type创建类理解什么是元类使用元类的意义元类实战:ORM . 类是如何产生的 类是如何产生?这个问题肯定很傻.实则不然,很多 ...
- [转] Python Traceback详解
追莫名其妙的bugs利器-mark- 转自:https://www.jianshu.com/p/a8cb5375171a Python Traceback详解 刚接触Python的时候,简单的 ...
- Python Collections详解
Python Collections详解 collections模块在内置数据结构(list.tuple.dict.set)的基础上,提供了几个额外的数据结构:ChainMap.Counter.deq ...
- python多线程详解
目录 python多线程详解 一.线程介绍 什么是线程 为什么要使用多线程 二.线程实现 threading模块 自定义线程 守护线程 主线程等待子线程结束 多线程共享全局变量 互斥锁 递归锁 信号量 ...
- Python数据类型详解——元组
Python数据类型详解--元组 有时候我们的列表数据不想被别人修改时该怎么办? 此时,就可以使用元组来存放,元祖又称为只读列表,不能修改 定义方式:与列表类似,将列表的[]换成()即可. 特性: 1 ...
- 【python】详解事件驱动event实现
[python]详解事件驱动event实现 所有的计算机程序都可以大致分为两类:脚本型(单次运行)和连续运行型(直到用户主动退出). 1.脚本型:脚本型的程序包括最早的批处理文件以及使用Python做 ...
随机推荐
- bootstrap 模态框(modal)插件使用
今天用户登陆时,在原网页上弹出新登陆窗口,发现使用的是modal插件,记录下该插件的使用方法,手写强化下. 首先,模态框(modal)是覆盖在父窗体上的子窗体,目的是显示来自一个单独的源的内容,可以在 ...
- ECEF和大地坐标系的相互转化
在阅读 RTKLIB的源码时,发现了ECEF和大地坐标系的相互转换的函数,大地坐标系(φ,λ,h)转成ECEF(X,Y,Z)与所看书籍(GPS原理与接收机,谢刚,电子工业出版社)的公式是一样的,而EC ...
- Java之排序
1.插入排序 假设第一个数已经是排好序的,把第二个根据大小关系插到第一个前面或维持不动,把第三个根据前面两个的大小关系插到对应位置,依次往后. public class InsertSort { pu ...
- C语言程序设计课程总结
第一次教授C语言程序设计课程,相比计算机组成原理.arm体系结构等偏向硬件的课程,C的教学方式要灵活一些.计算机组成原理课程偏向理论,哈尔滨工业大学的计算机组成原理是国家精品课,增加了mooc+spo ...
- C语言第十一次博客作业---函数嵌套调用
一.实验作业 1.1 PTA题目 题目:递归实现顺序输出整数 1. 本题PTA提交列表 2. 设计思路 printdigits函数 定义整型变量result存放结果 if n是10的倍数 result ...
- 团队作业7——第二次项目冲刺(Beta版本12.04)
1.当天站立式会议照片 本次会议内容:1:每个人汇报自己完成的工作.2:组长分配各自要完成的任务. 2.每个人的工作 黄进勇:项目整合,后台代码. 李勇:前台界面优化. 何忠鹏:数据库模块. 郑希彬: ...
- vue.js下载及安装配置
环境 Deepin15.4 下载及配置 node下载地址:http://nodejs.cn/download/ 解压到文件夹 /home/maskerk/vue/ 下 设置软连接: $ ln -s / ...
- 201621123068 Week02-Java基本语法与类库
1. 本周学习总结 1.1 当浮点数和整数放到一起运算时,java一般会将整数转化为浮点数然后进行浮点数计算,但是这样得出的结果通常与数学运算有一定误差,浮点数精确计算需要使用BigDecimal类 ...
- 201421123042 《Java程序设计》第11周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多线程相关内容. 2. 书面作业 本次PTA作业题集多线程 1. 源代码阅读:多线程程序BounceThread 1.1 BallR ...
- 《javascript设计模式与开发实践》阅读笔记(13)—— 职责链模式
职责链模式 使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系,将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止. 书里的订单的例子 假设我们负责一个售卖手机 ...