写在前面

之前在看廖雪峰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. 为 MaixPy 加入软 I2C 接口(移植 MicroPython 的 I2C)

    起因 本文的重心为讲解如何为一款芯片移植和实现 micropython 的通用组件,但会顺带解释不同芯片的工作方式和特性. 国际惯例,先有起因,再谈问题的解决,所以记得上次总结的 关于 K210 Ma ...

  2. springboot项目打包瘦身

    默认情况下,Spring Boot 项目发布时会将项目代码和项目的所有依赖文件一起打成一个可执行的 jar 包.但如果项目的依赖包很多,那么这个文件就会非常大.这样每次即使只改动一点东西,就需要将整个 ...

  3. LCD1602 库函数

    LCD1602 库函数 This library allows an Arduino board to control LiquidCrystal displays (LCDs) based on t ...

  4. vue+elmentUI项目的正则判断

    一.为了方便重复利用管理,我创建一个regExp.ts文件来管理正则的表达式,内容如下: 1 /* eslint-disable */ 2 const phoneNumberRegExp = /^[1 ...

  5. SpringBoot-06-模板引擎Thymeleaf

    6. 模板引擎 Thymeleaf Thyme leaf 英译为 百里香的叶子. 模板引擎 ​ 以前开发中使用的jsp就是一个模板引擎,但是springboot 以jar的方式,并且使用嵌入式的tom ...

  6. 用vscode写python,from引用本地文件的时候老是有红色波浪线,很不爽

    前言 出于一些原因,国际关系等等,最近想把开发工具切换到一些免费开源的工具上面,先尝试了在vscode上搭建python环境,总体还是很简单的, 网上教程很多,vscode本身的插件也很丰富,可惜了国 ...

  7. JVM 第五篇:命令行 JVM 故障处理工具

    本文内容过于硬核,建议有 Java 相关经验人士阅读. 1. 引言 虽然我们前面介绍了各种图形化 JVM 故障处理工具,但是很多情况下,我们在处理问题的时候并没有图形化的操作环境可以使用,这时候,就需 ...

  8. 代码上传多个git仓库,切换过remote后导致 can't update

    问题描述: 因为代码上传到github和coding 切换了 git--> rmote的地址:后来update失败 问题解决: 重新配置git解决:按提示操作就好 git fetch git p ...

  9. BST,Splay平衡树学习笔记

    BST,Splay平衡树学习笔记 1.二叉查找树BST BST是一种二叉树形结构,其特点就在于:每一个非叶子结点的值都大于他的左子树中的任意一个值,并都小于他的右子树中的任意一个值. 2.BST的用处 ...

  10. Jenkins集成appium自动化测试(Windows篇)

    一,引入问题 自动化测试脚本绝大部分用于回归测试,这就需要制定执行策略,如每天.代码更新后.项目上线前定时执行,才能达到最好的效果,这时就需要进行Jenkins集成. 不像web UI自动化测试可以使 ...