写在前面

之前在看廖雪峰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(元类)

  1. 先看下如何定义一个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
  2. 分析上面的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)
  3. 使用上面的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: 类中必须有文档注释
  4. 再测试下文档注释的限制
    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: 类中必须有文档注释
  5. 符合要求的类
    class Persion(metaclass=PersionMetaclass):
    """文档注释"""
    pass Persion()

参悟python元类(又称metaclass)系列实战(一)的更多相关文章

  1. python 元类 type metaclass

    python中一切皆对象,类对象创建实例对象,元类创建类对象,元类创建元类. 元类创建类对象有2中方式: 一.type方法 type(类名, 由父类名称组成的元组(针对继承的情况,可以为空),包含属性 ...

  2. python 元类(metaclass)

    元类参见老师的博客 http://www.cnblogs.com/linhaifeng/articles/8029564.html

  3. python元类:type和metaclass

    python元类:type和metaclass python中一切皆对象,所以类本身也是对象.类有创建对象的能力,那谁来创建类的呢?答案是type. 1.用tpye函数创建一个类 class A(ob ...

  4. 第十三章、元类(metaclass)

    目录 第十三章.元类(metaclass) 一.什么是元类 二.为什么用元类 第十三章.元类(metaclass) 一.什么是元类 在python中一切皆对象,那么我们用class关键字定义的类本身也 ...

  5. 深入理解python元类

    类也是对象 在理解元类之前,你需要先掌握Python中的类.Python 中的类概念借鉴 Smalltalk,这显得有些奇特.在大多数编程语言中,类就是一组用来描述如何生成一个对象的代码段.当然在 P ...

  6. Python 元类 - Metaclasses

    Python 元类 - Metaclasses 默认情况下儿, classes 是有 type() 构造的. 类的结构体在一个新的 namespace 被执行, 类的名字 class name 绑定( ...

  7. Python进阶丨如何创建你的第一个Python元类?

    摘要:通过本文,将深入讨论Python元类,其属性,如何以及何时在Python中使用元类. Python元类设置类的行为和规则.元类有助于修改类的实例,并且相当复杂,是Python编程的高级功能之一. ...

  8. Python 元类详解

    一.Type介绍 在Python中一切皆对象,类它也是对象,而元类其实就是用来创建类的对象(由于一切皆对象,所以元类其实也是一个对象). 先来看这几个例子: 例1: In [1]: type(12) ...

  9. python 元类metaclass

    文章转自:http://www.cnblogs.com/linhaifeng/articles/8029564.html 一 知识储备 exec:三个参数 参数一:字符串形式的命令 参数二:全局作用域 ...

随机推荐

  1. SFUD+FAL+EasyFlash典型场景需求分析,并记一次实操记录

    SFUD+FAL+EasyFlash典型场景需求分析:用整个flash存储数据,上千条数据,读取得时候用easyflash很慢,估计要检索整个flash太慢了. 改进方法:分区检索. 1存数据时,根据 ...

  2. SQL实战——01. 查找最晚入职员工的所有信息

    查找最晚入职员工的所有信息CREATE TABLE `employees` (`emp_no` int(11) NOT NULL,`birth_date` date NOT NULL,`first_n ...

  3. 生成器generator和迭代器Iterator

    一.列表生成式       在学习生成器迭代器之前先了解一下什么是列表生成式,列表生成式是Python内置的非常简单却强大的可以用来创建list的生成式.什么意思?举个例子,如果想生成列表[0,1,2 ...

  4. C1853 编译器错误:fatal error C1853: 'pjtname.pch' precompiled header file is from a previous

    转载:https://www.cnblogs.com/emanlee/archive/2010/10/16/1852998.html 用VC++ 2008 编写C语言程序,编译出现错误: 预编译头文件 ...

  5. Arduino - 串口操作函数与示例代码大全

    来源:https://blog.csdn.net/iracer/article/details/50334041 Arduino - 串口操作函数与示例代码大全 本文总结了Arduino常用串口操作函 ...

  6. 使用 PL/SQL Developer 导入 .sql 文件

    操作系统:Windows 10 x64 PL/SQL Developer Version 12.0.7.1837 (64 bit) 01.226959 第一节:下载 Oracle Database X ...

  7. Metasploit简单使用——后渗透阶段

    在上文中我们复现了永恒之蓝漏洞,这里我们学习一下利用msf简单的后渗透阶段的知识/ 一.meterperter常用命令 sysinfo #查看目标主机系统信息 run scraper #查看目标主机详 ...

  8. AD15使用笔记

    AD15使用笔记 1.板内孔开洞 步骤:选中图形->Tools->Convert->Creat Borad Cutout From Selected Primitives;

  9. ASP。NET MVC (NetCore 2.0)用于处理实体框架、DbContexts和对象的通用控制器和视图

    下载source - 1.5 MB 介绍 本文的源代码已更新到NetCore 2.0 ASP.净MVC项目. 当我们开始开发一个ASP.在Microsoft Visual Studio中,我们发现通过 ...

  10. 线程基本使用--Thread内部方法调用start

    一个问题,下面的代码会如何运行 public class TraditionalThread { public static void main(String[] args) { System.out ...