参悟python元类(又称metaclass)系列实战(一)
写在前面
之前在看廖雪峰python系列的教程时,对元类的章节一直头大,总在思考我到底适不适合学习python,咋这么难,尤其是ORM的部分,倍受打击;后来从0到1手撸了一套ORM,才稍微进阶了一点理解。
这个系列会从元类开始梳理,最后按廖老师的demo撸一个ORM,希望能对大家有所增益。
本文中提及的“模板”,泛指“类(class)”。
有误的地方恳请大神指正下。
先从普通类实例化的过程探究下
class Persion:
def __init__(self, name):
self.name = name
# 把它实例化
p1 = Persion()
print(type(p1)) # <class '__main__.Persion'>
# 打印的信息显示p1这个实例是由 Persion 这个模板创建的,那Persion这个模板是谁创建的?
print(type(Persion)) # <class 'type'>,是type创建了一个叫Persion的模板
# 由此可见,在实例化 p1 了时候,实际上是走了两步,1.用type创建Persion模板; 2.用Persion模板创建实例p1
思考1:对于过程1,是否就意味着可以用type代替class去创建Persion模板?
def fn(self, name):
self.name = name
# type 接收3个位置参数,1.名字:Str,2.父类们: tuple,3.绑定的属性(方法):dict
Persion = type('demo', (object,), dict(__init__=fn))
# 等同于前面通过class声明的写法
思考2:如果能自定义type的子类并用其创建Persion模板,就意味着可以定制创建过程,这种type的子类,就叫做metaclass(元类)
- 先看下如何定义一个type的子类
# 按照默认习惯,metaclass的类名总是以Metaclass结尾,以便清楚地表示这是一个metaclass
class PersionMetaclass(type):
# 重写父类type中的new方法
def __new__(cls, name: str, bases: tuple, attrs: dict):
"""
类似class中的__init__方法,实例化时被调用
@cls: 类似__init__中的self,代表自己,这里代指‘类’自己
@name: 模板的名字
@bases: 父类的集合
@attrs: 属性(方法)集
"""
# 调用type创建class
Persion = type.__new__(cls, name, bases, attrs)
return Persion
- 分析上面的code,不难发现在调type创建class之前,可以加入定制的内容
# 需求:1.类名首字母必须大写,2.类中必须有文档注释
class PersionMetaclass(type):
def __new__(cls, name, bases, attrs): if not name.istitle():
raise TypeError('类名首字母必须大写') cls.doc = attrs.get('__doc__')
if cls.doc is None or len(cls.doc.strip()) == 0:
raise TypeError('类中必须有文档注释') return type.__new__(cls, name, bases, attrs)
- 使用上面的metaclass测试一下类名首字母大写的限制
# 定义类的时候,需要显示的指出用 PersionMetaclass 来定制类,关键字“metaclass” class persion(metaclass=PersionMetaclass):
pass persion()
Traceback (most recent call last):
File "testmetaclass.py", line 39, in <module>
class Persion(metaclass=PersionMetaclass):
File "testmetaclass.py", line 35, in __new__
raise TypeError('类中必须有文档注释')
TypeError: 类中必须有文档注释
- 再测试下文档注释的限制
class Persion(metaclass=PersionMetaclass):
pass Persion()
Traceback (most recent call last):
File "testmetaclass.py", line 39, in <module>
class Persion(metaclass=PersionMetaclass):
File "testmetaclass.py", line 35, in __new__
raise TypeError('类中必须有文档注释')
TypeError: 类中必须有文档注释
- 符合要求的类
class Persion(metaclass=PersionMetaclass):
"""文档注释"""
pass Persion()
参悟python元类(又称metaclass)系列实战(一)的更多相关文章
- python 元类 type metaclass
python中一切皆对象,类对象创建实例对象,元类创建类对象,元类创建元类. 元类创建类对象有2中方式: 一.type方法 type(类名, 由父类名称组成的元组(针对继承的情况,可以为空),包含属性 ...
- python 元类(metaclass)
元类参见老师的博客 http://www.cnblogs.com/linhaifeng/articles/8029564.html
- python元类:type和metaclass
python元类:type和metaclass python中一切皆对象,所以类本身也是对象.类有创建对象的能力,那谁来创建类的呢?答案是type. 1.用tpye函数创建一个类 class A(ob ...
- 第十三章、元类(metaclass)
目录 第十三章.元类(metaclass) 一.什么是元类 二.为什么用元类 第十三章.元类(metaclass) 一.什么是元类 在python中一切皆对象,那么我们用class关键字定义的类本身也 ...
- 深入理解python元类
类也是对象 在理解元类之前,你需要先掌握Python中的类.Python 中的类概念借鉴 Smalltalk,这显得有些奇特.在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段.当然在 P ...
- Python 元类 - Metaclasses
Python 元类 - Metaclasses 默认情况下儿, classes 是有 type() 构造的. 类的结构体在一个新的 namespace 被执行, 类的名字 class name 绑定( ...
- Python进阶丨如何创建你的第一个Python元类?
摘要:通过本文,将深入讨论Python元类,其属性,如何以及何时在Python中使用元类. Python元类设置类的行为和规则.元类有助于修改类的实例,并且相当复杂,是Python编程的高级功能之一. ...
- Python 元类详解
一.Type介绍 在Python中一切皆对象,类它也是对象,而元类其实就是用来创建类的对象(由于一切皆对象,所以元类其实也是一个对象). 先来看这几个例子: 例1: In [1]: type(12) ...
- python 元类metaclass
文章转自:http://www.cnblogs.com/linhaifeng/articles/8029564.html 一 知识储备 exec:三个参数 参数一:字符串形式的命令 参数二:全局作用域 ...
随机推荐
- 你知道CPU结构也会影响Redis性能吗?
啦啦啦,我是卖身不卖艺的二哈,ε=(´ο`*)))唉错啦(我是开车的二哈),我又来了,铁子们一起开车呀! 今天来分析下CPU结构对Redis性能会有影响吗? 在进行Redis性能分析的时候,通常我们会 ...
- Oracle - ascii为0的陷阱
一.概述 ascii0是个空字符,如果将这个字符插入到oracle数据库中会是什么现象,是null吗? 二.正式实验 创建一张测试表 create table test(id int, name va ...
- ztree通过ajax加载json数据中文乱码的解决方法:springmvc配置
一.问题描述 使用zTree的异步刷新父级菜单时,服务器返回中文乱码,但项目中使用了SpringMvc,已经对中文乱码处理,为什么还会出现呢? 此处为的异步请求的配置: async: { enable ...
- python程序整理(1)
''' 用户登录验证 要求: 1. 系统⾃动⽣成4位随机数. 作为登录验证码. 直接用就好. 这里不用纠结 提示. 生成随机数的办法. from random import randint num = ...
- 【ST表】SCOI2016 萌萌哒
题目内容 洛谷链接 一个长度为\(n\)的大数,用\(S_1S_2S_3...S_n\)表示,其中\(S_i\)表示数的第\(i\)位,\(S_1\)是数的最高位,告诉你一些限制条件,每个条件表示为四 ...
- centos8平台redis cluster集群搭建(redis5.0.7)
一,规划 redis cluster 1,cluster采用六台redis,3主3从 redis1 : ip: 172.17.0.2 redis2 : ip: 172.17.0.3 red ...
- 使用Sparse Checkout 排除跟踪Git仓库中指定的目录或文件
应用场景 在一个大工程里包含由不同部门开发的模块时,项目的Git仓库肯定很大,造成每次Git操作相对比较耗时.因为开发人员一般只关心他们部门的模块的代码,所以完全可以排除一些他完全不需要用到的目录.这 ...
- Jmeter入门(3)- Jmeter录制脚本
一. 录制web端 1. Badboy的介绍和安装 1.1 使用第三方工具Badboy来录制. 免费的web自动化测试工具 一个浏览器模拟工具 主要进行脚本的录制和回访,和对录制脚本进行调试,可以将脚 ...
- STL: set和map的区别、联系、使用
set是一种关联式容器,其特性如下: set以RBTree作为底层容器 所得元素的只有key(键)没有value(值) 不允许出现键重复 所有的元素都会被自动排序 不能通过迭代器来改变set的值,因为 ...
- Ⅲ Dynamic Programming
Dictum: A man who is willing to be a slave, who does not know the power of freedom. -- Beck 动态规划(Dy ...