Python探索之Metaclass初步了解


Posted in Python onOctober 28, 2017

先以一个大牛的一段关于Python Metapgramming的著名的话来做开头:

Metaclasses are deeper magic than 99% of users should ever worry about. If you wonder whether you need them, you don't (the people who actually need them know with certainty that they need them, and don't need an explanation about why). ? Tim Peters

翻译一下:Metaclasses是99%的用户都无需费神的黑科技。如果你还在纠结你是不是需要它的话,答案是NO (真正需要的人根本不需要解释) ? Tim Peters

这是什么鬼话?道可道,非常道吗?

Meta?

好,装B已毕。这确实是一个冷僻的,不常用的话题。一篇短文肯定讲不完。 所以叫做初步了解。

python中的类

首先这里讨论的python类,都基于继承于object的新式类进行讨论。

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

class Trick(object):
  pass

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

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

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

input:
class Trick(object):
pass
print type('123')
print type(123)
print type(Trick())
output:
<type 'str'>
<type 'int'>
<class '__main__.Trick'>

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

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

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

英文meta这个词其实是从希腊语里面借来的。wikipedia上的解释是:

indicate a concept which is an abstraction behind another concept, used to complete or add to the latter

不看还好,其实看了更晕。好在后面的解释有一句“更高一层的抽象”,可以帮助理解。 其实我们可以这样理解。meta的意思就是“关于什么的什么”:比如metadata可以理解为“关于数据的数据”,metaprogramming可以理解为“关于编程的编程”。这就和“更高一层的抽象” 比较契合了。同时又隐隐和编程中的另一个永恒主题-recursion联系在了一起。

另外,meta这个词天朝这边翻译成“元”,海峡对岸翻译成“后设”。其实我都不大理解从何而来。

元类一般用于创建类。在执行类定义时,解释器必须要知道这个类的正确的元类。解释器会先寻找类属性__metaclass__,如果此属性存在,就将这个属性赋值给此类作为它的元类。如果此属性没有定义,它会向上查找父类中的__metaclass__.如果还没有发现__metaclass__属性,解释器会检查名字为__metaclass__的全局变量,如果它存在,就使用它作为元类。否则, 这个类就是一个传统类,并用 types.ClassType 作为此类的元类。

在执行类定义的时候,将检查此类正确的(一般是默认的)元类,元类(通常)传递三个参数(到构造器): 类名,从基类继承数据的元组,和(类的)属性字典。

实例

聚焦到我们今天的主题,metaprogramming就是编写用来生成代码的代码。

假设我们写了一个NB的函数,用来计算一个任意复杂的算数表达式的值:

像1+2, 3*6+10, 什么的都可以交给它去计算。这样的函数的算法不是我们的主题,所以我们请出python自带的大招eval(),一行就可以搞定了:

def calc(expression):
  return eval(expression)

因为输入的可能性是无限的,所以我们肯定要好好测试一下这个函数了。假定我们想了 上百个test case。又假定我们是用unittest这个module来做测试的。这样的测试程序一般会长成这样:

import unittest
class TestStringMethods(unittest.TestCase):
  def test_upper(self):
    self.assertEqual('foo'.upper(), 'FOO')
  def test_isupper(self):
    self.assertTrue('FOO'.isupper())
    self.assertFalse('Foo'.isupper())
  def test_split(self):
    s = 'hello world'
    self.assertEqual(s.split(), ['hello', 'world'])
    # check that s.split fails when the separator is not a string
    with self.assertRaises(TypeError):
      s.split(2)
if __name__ == '__main__':
  unittest.main()

所以我们的目的就是用metaprogramming的方式来自动产生类似上面的测试类。

先上程序后解释:

#!/usr/bin/python3
import unittest
def calc(expression):
  return eval(expression)
def add_test(name, asserts):
  def test_method(asserts):
    def fn(self):
      left, right = asserts.split('=')
      expected = str(calc(left))
      self.assertEqual(expected, right)
    return fn
  d = {'test1': test_method(asserts)}
  cls = type(name, (unittest.TestCase,), d)
  globals()[name] = cls
if __name__ == '__main__':
  for i, t in enumerate([
      "1+2=3",
      "3*5*6=90"]):
    add_test("Test%d" % i, t)
  unittest.main()

NB的calc()函数我们解释过了。main这段也比较简单:我们用声明的方式定义了一组测试,然后通过unittest来执行。

有点复杂的是add_test()。我们先来看看最内层的fn(self)这个方法。逻辑上,它就是把输入的测试用例分成两份,一份是calc()的输入,一份是我们期待的结果;然后调用calc(), 接着用assertEqual()来测试。

但是这个self有点奇怪 ? 这里没有类,哪里来的self? 其实fn(self)确实是一个类的方法,只不过这个类是我们通过代码动态生成的。也就是下面这一行:

cls = type(name, (unittest.TestCase,), d)

这里的type()就是通常我们用来检查某个变量的类型的那个函数。只不过它还有另外一种不大为人知的形式:

class type(name, bases, dict)

这第二种形式,就会产生一个新的类型。以我们的程序为例,就是以unit.TestCase为baseclass, 产生了一个名为TestN的新类型,改类型的实现由d给出,而d就包含了通过closure返回的fn(self)这个方法。只不过在这个新类里面,它的名字叫做 test1()。

最后,我们把这个新产生的类加入到当前全局符号表里面,也就相当于上面给出的unittest的例子。

所以,总结一下。当我们运行这个脚本的时候,这段比较短的代码会针对每一个测试的表达式产生一个新的测试类,并动态生成测试的方法加载到该类里面。unitest从globals中找到这些类并一一执行测试。

上面的例子中,其实一行一行手打calc(1+2) == 3也没什么大不了的。但是当你要表达的逻辑比较复杂的时候,metaprogramming的强大就体现出来了。

总结

那么,看完这篇文章,我们也成为Tim所说的1%的程序猿了!其实,也许他的意思是,99%的编程工作都用不到这样技巧。在一些特殊的场合,比如编写某种框架的时候,metaprogramming会做到事半功倍。祝你在实践中碰到这样的机会。

以上就是本文关于Python探索之Metaclass初步了解的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:Python编程之Re模块下的函数介绍、python中模块的__all__属性详解等,如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!

Python 相关文章推荐
编写Python脚本批量下载DesktopNexus壁纸的教程
May 06 Python
MySQL中表的复制以及大型数据表的备份教程
Nov 25 Python
使用PyInstaller将Python程序文件转换为可执行程序文件
Jul 08 Python
python虚拟环境的安装配置图文教程
Oct 20 Python
Python numpy中矩阵的基本用法汇总
Feb 12 Python
python异常处理、自定义异常、断言原理与用法分析
Mar 23 Python
Python selenium文件上传下载功能代码实例
Apr 13 Python
tensorflow实现将ckpt转pb文件的方法
Apr 22 Python
python库skimage给灰度图像染色的方法示例
Apr 27 Python
Pymysql实现往表中插入数据过程解析
Jun 02 Python
Python爬虫进阶之爬取某视频并下载的实现
Dec 08 Python
手把手教你使用TensorFlow2实现RNN
Jul 15 Python
Python编程之Re模块下的函数介绍
Oct 28 #Python
Python探索之静态方法和类方法的区别详解
Oct 27 #Python
Python探索之爬取电商售卖信息代码示例
Oct 27 #Python
Python 列表理解及使用方法
Oct 27 #Python
Python算法之求n个节点不同二叉树个数
Oct 27 #Python
Python探索之自定义实现线程池
Oct 27 #Python
python音频处理用到的操作的示例代码
Oct 27 #Python
You might like
PHP连接access数据库
2008/03/27 PHP
php5数字型字符串加解密代码
2008/04/24 PHP
PHP原生模板引擎 最简单的模板引擎
2012/04/25 PHP
ThinkPHP查询返回简单字段数组的方法
2014/08/25 PHP
thinkphp实现图片上传功能
2016/01/13 PHP
php并发加锁问题分析与设计代码实例讲解
2021/02/26 PHP
javascript 避免闭包引发的问题
2009/03/17 Javascript
asp.net 30分钟掌握无刷新 Repeater
2011/09/16 Javascript
网站404页面3秒后跳到首页的实例代码
2013/08/16 Javascript
jQuery中验证表单提交方式及序列化表单内容的实现
2014/01/06 Javascript
jQuery事件用法实例汇总
2014/08/29 Javascript
JavaScript中的值是按值传递还是按引用传递问题探讨
2015/01/30 Javascript
JavaScript AOP编程实例
2015/06/16 Javascript
jQuery+html5+css3实现圆角无刷新表单带输入验证功能代码
2015/08/21 Javascript
JQuery+Ajax实现数据查询、排序和分页功能
2015/09/27 Javascript
深入理解Java线程编程中的阻塞队列容器
2015/12/07 Javascript
小程序异步问题之多个网络请求依次执行并依次收集请求结果
2019/05/05 Javascript
vue element-ui实现input输入框金额数字添加千分位
2019/12/29 Javascript
[03:18]DOTA2亚洲邀请赛小组赛第一日 RECAP赛事回顾
2015/01/30 DOTA
[04:42]2015国际邀请赛CDEC战队晋级之路
2015/08/13 DOTA
[04:40]2016个国际邀请赛中国区预选赛场地——华西村观战指南
2016/06/25 DOTA
Python常用的日期时间处理方法示例
2015/02/08 Python
Python 3.x 新特性及10大变化
2015/06/12 Python
Python实现导出数据生成excel报表的方法示例
2017/07/12 Python
windows下python 3.6.4安装配置图文教程
2018/08/21 Python
Python3将jpg转为pdf文件的方法示例
2019/12/13 Python
python+selenium+Chrome options参数的使用
2020/03/18 Python
佐卡伊官网:中国知名珠宝品牌
2017/02/05 全球购物
夏洛特和乔治婴儿和儿童时装精品店:Charlotte and George
2018/06/06 全球购物
美国肌肉和力量商店:Muscle & Strength
2019/06/22 全球购物
英文简历自荐信范文
2013/12/11 职场文书
世博会口号
2014/06/20 职场文书
反腐倡廉主题教育活动总结
2015/05/07 职场文书
活动费用申请报告
2015/05/15 职场文书
Win10 和 Win11可以共存吗? win10/11产品生命周期/服务更新介绍
2021/11/21 数码科技
Java界面编程实现界面跳转
2022/06/16 Java/Android