python type 与 metaclass理解
简介
众所周知,type在一般情况下,我们都会去获取一个对象的类型,然后进行类型的比较;除此之外,type还有一个不为人知的作用:动态的创建类。在了解这个之前,首先了解以下type和isinstance之间的关系或者说是区别,这两个方法都可以判断类型,但又有所区别
type与isinstance
class Person:
def __init__(self) -> None:
pass
def sleep(self):
print("sleep")
class Male(Person):
def __init__(self) -> None:
pass
def sleep(self):
print("sleep")
print(type(Person))
print(type(Male))
print(isinstance(Person, type))
print(isinstance(Male, type))
p = Person()
m = Male()
print(type(p))
print(type(m))
print(type(p) == type(m))
print(isinstance(p, Person))
print(isinstance(m, Male))
print(isinstance(m, Person))
<class 'type'>
<class 'type'>
True
True
<class '__main__.Person'>
<class '__main__.Male'>
False
True
True
True
由结果<class 'type'>可以看出,所有的类默认是继承自type类,也就是说type默认是所有类的元类;由isinstance与type之间的对比来看,isinstance会考虑到继承关系,但是type不会考虑到继承关系,通俗点讲就是,对于子类与父类的对象类型比较时,isinstance会认为是相同的,但是type会认为不是相同的。
因此建议在比较对象类型时最好使用isinstance方法。
如何动态创建一个类?
由上述可知,type默认是所有类的元类,因此我们可以使用type创建一个简单的类:
type(__name: str, __bases: tuple[type, ...], __dict: dict[str, Any])
- __name:类的名称
- __bases:继承的类,以元组形式,注意单个元素时需要在后面添加逗号
- __dict:类中的属性或者方法
详细代码如下:
def run(self):
print("run"
class_name = "Test"
base_name = object
name = "tom"
Person2 = type(
class_name,
(base_name,),
{"run": run, "name": name},
)
print(type(Person2))
p = Person2()
print(type(p))
p.run()
<class 'type'>
<class '__main__.Test'>
run
metaclass作用
当我们需要对一种类型的类批量添加某些属性或者方法时,使用metaclass创建元类的派生类,从而添加属性或者方法
class FirstMetaClass(type):
# cls代表动态修改的类
# name代表动态修改的类名
# bases代表被动态修改的类的所有父类
# attr代表被动态修改的类的所有属性、方法组成的字典
def __new__(cls, name, bases, attrs):
# 动态为该类添加一个name属性
attrs['name'] = "C语言中文网"
attrs['say'] = lambda self: print("调用 say() 实例方法")
return super().__new__(cls, name, bases, attrs)
class Test(metaclass=FirstMetaClass):
def __init__(self) -> None:
pass
def __call__(self, *args, **kwds):
pass
t = Test()
print(t.name)
t.say()
C语言中文网
调用 say() 实例方法
由上述看出,当指定了元类后,添加的属性可以直接调用,类似于子类可以直接调用父类的方法一样,其实元类就相当于时每个类的父类,因此修改元类属性或者方法,可以影响到子类,父类与子类的关系和元类与普通类的关系是类似的。

python type 与 metaclass理解的更多相关文章
- python——type()、metaclass元类和精简ORM框架
1.type()函数 if __name__ == '__main__': h = hello() h.hello() print(type(hello)) print(type(h)) Hello, ...
- Python type class metaclass
'type' 是 python built-in metaclass 其他继承自 ‘type’的class都可以是 Metaclass 子类可以继承父类的metaclass 然而 __metaclas ...
- python元类:type和metaclass
python元类:type和metaclass python中一切皆对象,所以类本身也是对象.类有创建对象的能力,那谁来创建类的呢?答案是type. 1.用tpye函数创建一个类 class A(ob ...
- python 元类——metaclass
from stack overflow:http://stackoverflow.com/questions/100003/what-is-a-metaclass-in-python Classes ...
- Python Type Hint类型注解
原文地址:https://realpython.com/python-type-checking/ 在本指南中,你将了解Python类型检查.传统上,Python解释器以灵活但隐式的方式处理类型.Py ...
- Python中文字符的理解:str()、repr()、print
Python中文字符的理解:str().repr().print 字数1384 阅读4 评论0 喜欢0 都说Python人不把文字编码这块从头到尾.从古至今全研究通透的话是完全玩不转的.我终于深刻的理 ...
- Python type类具体的三大分类:metaclasses,classes,instance
Python type类视角中的对象体系需要我们不断的学习,其中我们使用的时候需要注意.下面我们就看看如何才能更好的运用Python type类.下面的文章希望大家有所收获. 在单纯的Python t ...
- python 导入模块 import 理解
--python 导入模块 import 理解 -----------------------------------2014/03/18 python 导入一个模块的过程要求有一个叫做“路径搜索”的 ...
- 【python进阶】深入理解系统进程2
前言 在上一篇[python进阶]深入理解系统进程1中,我们讲述了多任务的一些概念,多进程的创建,fork等一些问题,这一节我们继续接着讲述系统进程的一些方法及注意点 multiprocessing ...
随机推荐
- java、selenium、图片滑块验证,底部附本地可测试代码
准备 本地Chrome版本对应WebDriver驱动:http://chromedriver.storage.googleapis.com/index.html maven包 <!-- sele ...
- django基础--02基于数据库的小项目
摘要:简单修改.增加部分页面,了解django开发的过程.(Python 3.9.12,django 4.0.4 ) 接前篇,通过命令: django-admin startproject myWeb ...
- Jmeter接口测试流程详解(中科软测认证中心)
1.jmeter简介 Jmeter是由Apache公司开发的java开源项目,所以想要使用它必须基于java环境才可以: Jmeter采用多线程,允许通过多个线程并发取样或通过独立的线程对不同的功能同 ...
- 【Azure 环境】【Azure Developer】使用Python代码获取Azure 中的资源的Metrics定义及数据
问题描述 使用Python SDK来获取Azure上的各种资源的Metrics的名称以及Metrics Data的示例 问题解答 通过 azure-monitor-query ,可以创建一个 metr ...
- 跨云平台与物理专线使用Vxlan实现两地二层互通,并使用ospf与bgp做底层链路主备
Vxlan基础,已掌握可略过 VXLAN网络架构 VXLAN是NVO3中的一种网络虚拟化技术,通过将原主机发出的数据包封装在UDP中,并使用物理网络的IP.MAC作为外层头进行封装,然后在IP网络上传 ...
- vue大型电商项目尚品汇(前台篇)day02
现在正式回归,开始好好做项目了,正好这一个项目也开始慢慢的开始起色了,前面的准备工作都做的差不多了. 而且我现在也开始慢慢了解到了一些项目才开始需要的一些什么东西了,vuex.router这些都是必备 ...
- Vagrant之CentOS
Vagrant之CentOS Vagrant官网 https://www.vagrantup.com https://app.vagrantup.com/boxes/search https://ap ...
- 配置中心Nacos(服务发现)
服务演变之路 单体应用架构 在刚开始的时候,企业的用户量.数据量规模都⽐较⼩,项⽬所有的功能模块都放在⼀个⼯程中编码.编译.打包并且部署在⼀个Tomcat容器中的架构模式就是单体应用架构,这样的架构既 ...
- 绿色城市智慧运营:Web 3D 垃圾分类 GIS 系统
前言 感谢所有为上海疫情奉献的人,祈求上海疫情早日清零,中国加油,上海加油! <上海市生活垃圾管理条例>施行至今已有两年多,上海市民践行绿色低碳理念.主动参与生活垃圾分类的习惯基本养成,分 ...
- Kubernetes client-go 源码分析 - ListWatcher
概述ListWatch 对象的创建GetterListWatchList() & Watch() 概述 源码版本信息 Project: kubernetes Branch: master La ...