Python类和元类(metaclass)的理解和简单运用

简介:


(一) python中的类

今天看到一篇好文,然后结合自己的情况总结一波。

这里讨论的python类,都基于python2.7x以及继承于object的新式类进行讨论。

首先在python中,所有东西都是对象。这句话非常重要要理解元类我要重新来理解一下python中的类。


 
 
  1. class Trick(object): 
  2.     pass 

当python在执行带class语句的时候,会初始化一个类对象放在内存里面。例如这里会初始化一个Trick对象。

这个对象(类)自身拥有创建对象(通常我们说的实例,但是在python中还是对象)的能力。

为了方便后续理解,我们可以先尝试一下在新式类中最古老厉害的关键字type。


 
 
  1. input: 
  2. class Trick(object): 
  3. pass 
  4.  
  5. print type('123'
  6. print type(123) 
  7. print type(Trick()) 
  8.  
  9. output
  10. <type 'str'
  11. <type 'int'
  12. <class '__main__.Trick'>  

可以看到能得到我们平时使用的 str, int, 以及我们初始化的一个实例对象Trick()

但是下面的方法你可能没有见过,type同样可以用来动态创建一个类

type(类名, 父类的元组(针对继承的情况,可以为空),包含属性的字典(名称和值))

这个怎么用呢,我要用这个方法创建一个类 让我们看下下面的代码


 
 
  1. input: 
  2. print type('trick', (), {}) 
  3.  
  4. output
  5. <class '__main__.trick'
  6. 同样我们可以实例化这个类对象 
  7.  
  8. input: 
  9. print type('trick', (), {})() 
  10.  
  11. output
  12. <__main__.trick object at 0x109283450>  

可以看到,这里就是一个trick的实例对象了。

同样的这个方法还可以初始化创建类的父类,同时也可以初始化类属性:


 
 
  1. input: 
  2. class FlyToSky(object): 
  3.     pass 
  4.  
  5. pw = type('Trick', (FlyToSky, ), {'laugh_at''hahahaha'}) 
  6. print pw().laugh_at 
  7. print pw.__dict__ 
  8. print pw.__bases__ 
  9. print pw().__class__ 
  10. print pw().__class__.__class__ 
  11.  
  12.  
  13. output
  14. hahahaha 
  15. {'__module__''__main__''laugh_at''hahahaha''__doc__': None} 
  16. (<class '__main__.FlyToSky'>,) 
  17. <class '__main__.Trick'
  18. <type 'type'>  

下面我将依次理一下上面的内容,在此之前我必须先介绍两个魔法方法:

  1. __class__这个方法用于查看对象属于是哪个生成的,这样理解在python中的所有东西都是对象,类对象也是对象。如果按照以前的思维来想的话就是类是元类的实例,而实例对象是类的实例。
  2. __bases__这个方法用于得到一个对象的父类是谁,特别注意一下__base__返回单个父类,__bases__以tuple形式返回所有父类。

好了知道了这两个方法我来依次说一下每行什么意思。

  1. 使用type创建一个类赋值给pw type的接受的三个参数的意思分辨是(类的名称, 类是否有父类(), 类的属性字典{})
  2. 这里初始化一个类的实例,然后尝试去获得父类的laugh_at属性值,然后得到结果hahahaha
  3. 取一个pw的也就是我们常见类的类字典数据
  4. 拿到pw的父类,结果是我们指定的 FlyToSky
  5. pw的实例pw()属于哪个类初始化的,可以看到是class Trick
  6. 我们再看class trick是谁初始化的? 就是元类type了

(二) 什么是元类以及简单运用

这一切介绍完之后我们总算可以进入正题

到底什么是元类?通俗的就是说,元类就是创建类的类。。。这样听起来是不是超级抽象?

来看看这个


 
 
  1. Trick = MetaClass() 
  2. MyObject = Trick()  

上面我们已经介绍了,搞一个Trick可以直接这样


 
 
  1. Trick = type('Trick', (), {}) 

可以这样其实就是因为,Type实际上是一个元类,用他可以去创建类。什么是元类刚才说了,元类就是创建类的类。也可以说他就是一个类的创建工厂。

类上面的__metaclass__属性,相信愿意了解元类细节的盆友,都肯定见过这个东西,而且为之好奇。不然我不知道是什么支撑你看到这里的。使用了__metaclass__这个魔法方法就意味着就会用__metaclass__指定的元类来创建类了。


 
 
  1. class Trick(FlyToSky): 
  2.     pass  

当我们在创建上面的类的时候,python做了如下的操作:

Trick中有__metaclass__这个属性吗?如果有,那么Python会在内存中通过__metaclass__创建一个名字为Trick的类对象,也就是Trick这个东西。如果Python没有找到__metaclass__,它会继续在自己的父类FlyToSky中寻找__metaclass__属性,并且尝试以__metaclass__指定的方法创建一个Trick类对象。如果Python在任何一个父类中都找不到__metaclass__,它也不会就此放弃,而是去模块中搜寻是否有__metaclass__的指定。如果还是找不到,好吧那就是使用默认的type来创建Trick。

那么问题来了,我们要在__metaclass__中放置什么呢?答案是可以创建一个类的东西,type,或者任何用到type或子类化type的东西都行。

(三) 自定义元类

自定义类的的目的,我总结了一下就是拦截类的创建,然后修改一些特性,然后返回该类。是不是有点熟悉?没错,就是感觉是装饰器干的事情,只是装饰器是修饰一个函数,同样是一个东西进去,然后被额外加了一些东西,最后被返回。

其实除了上面谈到的制定一个__metaclass__并不需要赋值给它的不一定要是正式类,是一个函数也可以。要创建一个使所有模块级别都是用这个元类创建类的话,在模块级别设定__metaclass__就可以了。先写一个来试试看,我还是延用stackoverflow上面那个哥们的例子,将所有的属性都改为大写的。🤗

来看这个例子:


 
 
  1. input: 
  2. def upper_attr(class_name, class_parents, class_attr): 
  3.     ""
  4.     返回一个对象,将属性都改为大写的形式 
  5.     :param class_name:  类的名称 
  6.     :param class_parents: 类的父类tuple 
  7.     :param class_attr: 类的参数 
  8.     :return: 返回类 
  9.     ""
  10.     # 生成了一个generator 
  11.     attrs = ((name, value) for name, value in class_attr.items() if not name.startswith('__')) 
  12.     uppercase_attrs = dict((name.upper(), value) for name, value in attrs) 
  13.     return type(class_name, class_parents, uppercase_attrs) 
  14.  
  15. __metaclass__ = upper_attr 
  16.  
  17. pw = upper_attr('Trick', (), {'bar': 0}) 
  18. print hasattr(pw, 'bar'
  19. print hasattr(pw, 'BAR'
  20. print pw.BAR 
  21.  
  22. output
  23. False 
  24. True 
  25. 0  

可以从上面看到,我实现了一个元类(metaclass), 然后指定了模块使用这个元类来创建类,所以当我下面使用type进行类创建的时候,可以发现小写的bar参数被替换成了大写的BAR参数,并且在最后我调用了这个类属性并,打印了它。

上面我们使用了函数做元类传递给类,下面我们使用一个正式类来作为元类传递给__metaclass__


 
 
  1. class UpperAttrMetaClass(type): 
  2.     def __new__(mcs, class_name, class_parents, class_attr): 
  3.         attrs = ((name, value) for name, value in class_attr.items() if not name.startswith('__')) 
  4.         uppercase_attrs = dict((name.upper(), value) for name, value in attrs) 
  5.         return super(UpperAttrMetaClass, mcs).__new__(mcs, class_name, class_parents, uppercase_attrs) 
  6.  
  7.  
  8. class Trick(object): 
  9.     __metaclass__ = UpperAttrMetaClass 
  10.     bar = 12 
  11.     money = 'unlimited' 
  12.  
  13. print Trick.BAR 
  14. print Trick.MONEY  

总结:

啊好累好累终于写完了。。。写了好久,总之就像我上面说的,略带一点装饰器的思路去理解元类这件事情,可能会让你豁然开朗。元类这种黑暗魔法按照常理来说是不应该被广泛使用的,从写业务代码一年差不多一年,除了在完成kepler项目的时候稍微黑魔法了一下(实际是根本不需要这样操作),其他地方都没有用到过。等到真正需要的时候,你可能不会去思考为什么要去使用,而是因为要解决问题所以就是要这样写,所以才出现了元类这种东西。我是这样理解的,一个东西存在的真正意义就在于你可以用这个东西去解决以前难以解决的问题,可以让难以解决的问题变得简单起来,而不是为了炫技让一个问题变得复杂起来。


作者:piperck

来源:51CTO

相关文章
|
1天前
|
测试技术 Python
【03】做一个精美的打飞机小游戏,规划游戏项目目录-分门别类所有的资源-库-类-逻辑-打包为可玩的exe-练习python打包为可执行exe-优雅草卓伊凡-持续更新-分享源代码和游戏包供游玩-1.0.2版本
【03】做一个精美的打飞机小游戏,规划游戏项目目录-分门别类所有的资源-库-类-逻辑-打包为可玩的exe-练习python打包为可执行exe-优雅草卓伊凡-持续更新-分享源代码和游戏包供游玩-1.0.2版本
【03】做一个精美的打飞机小游戏,规划游戏项目目录-分门别类所有的资源-库-类-逻辑-打包为可玩的exe-练习python打包为可执行exe-优雅草卓伊凡-持续更新-分享源代码和游戏包供游玩-1.0.2版本
|
22天前
|
数据采集 存储 XML
python实战——使用代理IP批量获取手机类电商数据
本文介绍了如何使用代理IP批量获取华为荣耀Magic7 Pro手机在电商网站的商品数据,包括名称、价格、销量和用户评价等。通过Python实现自动化采集,并存储到本地文件中。使用青果网络的代理IP服务,可以提高数据采集的安全性和效率,确保数据的多样性和准确性。文中详细描述了准备工作、API鉴权、代理授权及获取接口的过程,并提供了代码示例,帮助读者快速上手。手机数据来源为京东(item.jd.com),代理IP资源来自青果网络(qg.net)。
|
3月前
|
索引 Python
python-类属性操作
【10月更文挑战第11天】 python类属性操作列举
37 1
|
3月前
|
Java C++ Python
Python基础---类
【10月更文挑战第10天】Python类的定义
33 2
WK
|
3月前
|
Python
Python类命名
在Python编程中,类命名至关重要,影响代码的可读性和维护性。建议使用大写驼峰命名法(如Employee),确保名称简洁且具描述性,避免使用内置类型名及单字母或数字开头,遵循PEP 8风格指南,保持项目内命名风格一致。
WK
30 0
|
3月前
|
开发者 Python
Python中的元类深度剖析与实战应用
Python中的元类深度剖析与实战应用
51 0
|
3月前
|
程序员 开发者 Python
深度解析Python中的元编程:从装饰器到自定义类创建工具
【10月更文挑战第5天】在现代软件开发中,元编程是一种高级技术,它允许程序员编写能够生成或修改其他程序的代码。这使得开发者可以更灵活地控制和扩展他们的应用逻辑。Python作为一种动态类型语言,提供了丰富的元编程特性,如装饰器、元类以及动态函数和类的创建等。本文将深入探讨这些特性,并通过具体的代码示例来展示如何有效地利用它们。
70 0
|
3月前
|
Python
Python中的类(一)
Python中的类(一)
26 0
|
3月前
|
Python
Python中的类(一)
Python中的类(一)
21 0
|
3月前
|
Python
Python中的类(二)
Python中的类(二)
28 0