Python 元类详解
一、Type介绍
在Python中一切皆对象,类它也是对象,而元类其实就是用来创建类的对象(由于一切皆对象,所以元类其实也是一个对象)。
先来看这几个例子:
例1:
In [1]: type(12)
Out[1]: int
通过 type 可以查看对象的类型,也就是查看对象是那一类的,这里可以看出来 12 是 int 类型的也就是整数类。由于一切皆对象,那么我们也可以查看 int 是属于哪一个类型的。
In [2]: type(int)
Out[2]: type
通过这样的操作可以看出来 int 是 type 类型的,也就是说 int 类是 type 类实例化得到的对象。按照这样的思路在来看看 type 类是什么类型。
In [3]: type(type)
Out[3]: type
从上面可以看出来 type 是 type 类型的对象
例2:在来看这样一个规律
In [4]: class Foo:
...: pass
...:
In [5]: f = Foo()
In [6]: type(f)
Out[6]: __main__.Foo
In [7]: type(Foo)
Out[7]: type
In [8]: type(type)
Out[8]: type
由上可以看出,实例 f 是 Foo 类的对象,Foo 这个自定义的类它是 type 类型的,并且 type 也是 type 类型的。
总结:
- 实例是类的实例;----------------------> f 是 Foo 类的实例
- 类是 type 类的实例;-----------------> Foo 类是 type 类的实例
- type 类是 type 类的实例;
从上面的规律可以看出,所有的类终归都是由 type 实例化得来的,它就是最元始的,同时它也是一个类,所以也可以称之为元类(Metaclass)
实例、类、元类之间的关系如图所示:

二、Type 使用方法
在 help(type) 显示的帮助文档中,可以看到这样两种使用方法;
class type(object)
| type(object) -> the object's type
| type(name, bases, dict, **kwds) -> a new type
第一种:直接带入对象,返回对象的类型,也是我们最常用的。
第二种:type(name, bases, dict, **kwds) 返回一个新的类型,也就是一个新的类。
- name:新类的名称;
- bases:以元组的形式,声明父类;
- dict:以字典的形式声明类的属性;
实例1:用 type 创建一个名为 Foo1 的类
In [10]: Foo1 = type("Foo1", (), {})
In [11]: Foo1.__bases__
Out[11]: (object,)
虽然没有给 bases 赋值,但是 Python 中所有的类都是以 object 为基类,所以查看类 Foo1 的父类,会得到这样一个结果。PS:
__bases__查看父类。
也可以将 Foo1 实例化,就能看见 f 是 Foo1 类型,并且仿照之前的样子查找它的类型,可以一直查找到元类 type。
In [12]: f = Foo1()
In [13]: f.__class__
Out[13]: __main__.Foo1
In [14]: f.__class__.__class__
Out[14]: type
__class__是类的一个内置属性,表示类的类型,也是类的实例的属性,表示实例对象的类
实例2:利用元类 type 所创建的类 Bar 继承了 Foo1 和 Book 两个类
In [15]: class Book:
...: def __new__(cls, *args, **kwargs):
...: cls.website = 'www.cnblogs.com'
...: return super().__new__(cls)
...:
In [16]: Bar = type('Bar', (Foo1, Book), {'name': 'python'})
In [17]: b = Bar()
In [18]: b.name
Out[18]: 'python'
In [19]: b.website
Out[19]: 'www.cnblogs.com'
In [20]: b.__class__
Out[20]: __main__.Bar
In [21]: b.__class__.__class__
Out[21]: type
In [22]: Bar.__bases__
Out[22]: (__main__.Foo1, __main__.Book)
In [23]: Bar.__dict__
Out[23]: mappingproxy({'name': 'python',
'__module__': '__main__',
'__doc__': None,
'website': 'www.cnblogs.com'})
上述中 Bar 类除了在创建时设置的 name 外,还有从 Book 类继承来的 website,实例化后也可以读取到 name 和 website 两个属性的值,并且实例的类型是 Bar。
三、Metaclass
虽然使用第二种方法能定义出所需要的各种类型的类,但是,在实践中,这种形式并不常用,而是用这种形式;
In [24]: class Meta(type):
...: pass
...:
In [25]: class Spam(metaclass=Meta):
...: pass
...:
In [26]: s = Spam()
类 Meta 继承了 type ,也可以称为元类,而 Spam 类与之前定义的类的不同之处就在于,类名后括号内添加了参数 metaclass 来说明或指定元类,这样就利用了元类 Meta 创建了类 Spam。
总结描述:
- 类 Spam 是元类 Meta 的实例;
- 元类 Meta 实例化得到了类 Spam;
In [27]: s.__class__
Out[27]: __main__.Spam
In [28]: s.__class__.__class__
Out[28]: __main__.Meta
In [29]: s.__class__.__class__.__class__
Out[29]: type
下面做一个更详细的实例,来了解 Spam 与 Meta 之间的关系。
class Meta(type):
def __new__(cls, name, bases, attrs):
print("执行了元类 Meta!")
attrs['author'] = "xiaoyang-sir"
spam_class = super().__new__(cls, name, bases, attrs)
print(spam_class)
return spam_class
class Spam(metaclass=Meta):
def __init__(self, website):
self.website = website
print(Spam.author)
"""
Out:
执行了元类 Meta!
<class '__main__.Spam'>
xiaoyang-sir
"""
从上面程序运行的的结果可以直接看出当该段程序运行的时候,虽然并没有直接实例化 Spam 类,但是由于指定了 Meta 为 Spam 类的元类 ,所以 Meta 会直接实例化 Spam 类,并且元类 Meta 的构造方法
__new__()中定义的属性 author 已经成为了实例 Spam 类的类属性。
那么接下来将 Spam 实例化后,也就能得到想要的结果。
s = Spam('www.cnblogs.com')
print(s.website)
print(s.author)
"""
Out:
执行了元类 Meta!
<class '__main__.Spam'>
www.cnblogs.com
xiaoyang-sir
"""
四、元类实现单例模式
元类实现单例模式可以在元类中定义 __call__() 方法,__call__() 方法主要是在对象加括号被调用的时候执行 __call__() 方法中的内容。由于类是一个对象,所以在类加括号的时候就会执行元类中的__call__()方法。
class Meta(type):
__instance = None
def __call__(cls, *args, **kwargs):
if not cls.__instance:
cls.__instance = type.__call__(cls, *args, **kwargs)
return cls.__instance
class Spam(metaclass=Meta):
pass
x = Spam()
y = Spam()
print(x)
print(y)
print(x==y)
"""
Out:
<__main__.Spam object at 0x00000251A51E7E50>
<__main__.Spam object at 0x00000251A51E7E50>
True
"""
Python 元类详解的更多相关文章
- Python之元类详解
一.引子 元类属于Python面向对象编程的深层魔法,99%的人都不得要领,一些自以为搞明白元类的人其实也是自圆其说,点到为止,从队元类的控制上来看就破绽百出,逻辑混乱: 二.什么是元类 一切源自于一 ...
- python定制类详解
1.什么是定制类python中包含很多内置的(Built-in)函数,异常,对象.分别有不同的作用,我们可以重写这些功能. 2.__str__输出对象 class Language(object): ...
- 【python进阶】详解元类及其应用2
前言 在上一篇文章[python进阶]详解元类及其应用1中,我们提到了关于元类的一些前置知识,介绍了类对象,动态创建类,使用type创建类,这一节我们将继续接着上文来讲~~~ 5.使⽤type创建带有 ...
- Python基础知识详解 从入门到精通(七)类与对象
本篇主要是介绍python,内容可先看目录其他基础知识详解,欢迎查看本人的其他文章Python基础知识详解 从入门到精通(一)介绍Python基础知识详解 从入门到精通(二)基础Python基础知识详 ...
- Python 多进程 multiprocessing.Pool类详解
Python 多进程 multiprocessing.Pool类详解 https://blog.csdn.net/SeeTheWorld518/article/details/49639651
- Python 字符串方法详解
Python 字符串方法详解 本文最初发表于赖勇浩(恋花蝶)的博客(http://blog.csdn.net/lanphaday),如蒙转载,敬请保留全文完整,切勿去除本声明和作者信息. ...
- (转)python collections模块详解
python collections模块详解 原文:http://www.cnblogs.com/dahu-daqing/p/7040490.html 1.模块简介 collections包含了一些特 ...
- Python字符编码详解,str,bytes
什么是明文 “明文”是可以是文本,音乐,可以编码成mp3文件.明文可以是图像的,可以编码为gif.png或jpg文件.明文是电影的,可以编码成wmv文件.不一而足. 什么是编码?把明文变成计算机语言 ...
- python自定义异常实例详解
python自定义异常实例详解 本文通过两种方法对Python 自定义异常进行讲解,第一种:创建一个新的exception类来拥有自己的异常,第二种:raise 唯一的一个参数指定了要被抛出的异常 1 ...
随机推荐
- 改善java程序
1.用偶判断,不用奇判断.因为负数会出错. // 不使用 String str = i + "->" + (i%2 == 1? "奇数": "偶 ...
- 羽夏 Bash 简明教程(下)
写在前面 该文章根据 the unix workbench 中的 Bash Programming 进行汉化处理并作出自己的整理,并参考 Bash 脚本教程 和 BashPitfalls 相关内容 ...
- 【Java8新特性】Stream(分类+案例)
一.Stream概述 什么是Stream? Stream是Java8引入的全新概念,它用来处理集合中的数据,可以让你以一种声明的方式处理数据. Stream 使用一种类似用 SQL 语句从数据库查询数 ...
- 基于Python的渗透测试信息收集系统的设计和实现
信息收集系统的设计和实现 渗透测试是保卫网络安全的一种有效且必要的技术手段,而渗透测试的本质就是信息收集,信息搜集整理可为后续的情报跟进提供强大的保证,目标资产信息搜集的广度,决定渗透过程的复杂程度, ...
- PostgreSQL 的窗口函数 OVER, WINDOW, PARTITION BY, RANGE
最近在数据处理中用到了窗函数, 把使用方法记录一下, 暂时只有分组排序和滑动时间窗口的例子, 以后再逐步添加 场景 在SQL查询时, 会遇到有两类需要分组统计的场景, 在之前的SQL语法中是不方便实现 ...
- 面试官:Netty心跳检测机制是什么,怎么自定义检测间隔时间?
哈喽!大家好,我是小奇,一位热爱分享的程序员 小奇打算以轻松幽默的对话方式来分享一些技术,如果你觉得通过小奇的文章学到了东西,那就给小奇一个赞吧 文章持续更新 一.前言 书接上回,昨天在地里干了一天的 ...
- Docker权限 “Got permission denied while trying to connect to the Docker daemon socket at unix:///var/”
问题及解决办法 在普通用户下执行docker命令需要用sudo,没加sudo出现了下图所示的提示: 从上图看出,权限不足连接/var/run/docker.sock,我们看下这个文件: 可以看出,这个 ...
- 毕设(1)——机械臂DH建模
目录 毕设(1)--机械臂DH建模 改进DH参数表 Matlab代码验证 毕设中用到了很多代码,其中一部分我通过看书和看论文学习并实现的代码,会通过Gitee仓库分享出来,这些代码仅用于学习使用,祝各 ...
- Redis - 为什么 Redis 是单线程的?
Redis中work线程是单线程的.也就是对于业务数据的操作是单线程的. Redis中存在多线程操作 异步关闭文件 异步将缓冲区冲洗到磁盘文件中 异步删除键值对 Redis是基于内存的,所以cpu不是 ...
- Amazon 消息订阅对接
亚马逊的api 谁用谁知道...... 除了坑还是坑 头疼一周整出来,分享给铁汁们 amazon 的订阅思维,我只能说外国人脑回路有点长 下面就讲讲具体流程步骤: 第一步: 参照官方教程:设置通知(A ...