Python超详细讲解元类的使用

 更新时间:2022年06月28日 10:19:18   作者:henry_rhy  
在Python里一切都是对象(object),基本数据类型,如数字,字符串,函数都是对象。对象可以由类(class)进行创建。那么既然一切都是对象,那么类是对象吗?是的,类也是对象,那么又是谁创造了类呢?答案也很简单,也是类,一个能创作类的类,称之为(type)元类

类的定义

对象是通过类创建的,如下面的代码:

# object 为顶层基类
class Work(object):
    a = 100
Mywork = Work()   # 实例化
print(Mywork )   # Mywork 是 Work 所创建的一个对象  <__main__.Work object at 0x101eb4630>
print(type(Mywork))    # <class '__main__.Work'>
print(type(Work))    # 类型为元类 <class 'type'>

解析:

可以看见对象 Mywork 是类 Work 创建的实例。然但是可以看到Work的类型时由一个叫 type 的类创建的实例。即 Mywork —》 Work —》 type 创建

上面的例子中对象是动态创建的,类则是通过关键字 class 声明定义的

那么class关键字背后的玄机是什么呢?

  • 实际上,class Work(object) 这样的代码,等价于 Work = type(‘Work’, (objects, ), {“a”:100})
  • 即类 type 通过实例化创建了它的对象 Work,而这个 Work 恰恰是一个类,这样能创建类的类,就是 Python 的元类。而在python中内置的元类叫:type

一、什么是元类

  • 用来创建类的类,叫做元类
  • 类是元类创建出来的对象
  • 函数type实际上是一个元类,type就是Python在背后用来创建所有类的元类

二、注意区分元类和继承的基类

  • type是元类,所有的类都是通过type所创建出来的
  • object顶层的基类,所有类的继承顶层父类都是object

三、type 元类的使用

可以看到type是小写,一般情况下都会认为它是一个函数,通过查看源代码去看下如下定义的:

class type(object):
"""
type(object_or_name, bases, dict)
type(object) -> the object's type
type(name, bases, dict) -> a new type
"""
	# 实例化
	def __init__(cls, what, bases=None, dict=None): # known special case of type.__init__
	     """
	     type(object_or_name, bases, dict)
	     type(object) -> the object's type
	     type(name, bases, dict) -> a new type
	     # (copied from class doc)
	     """
	     pass
	# 创建类    
	@staticmethod     # known case of __new__
    def __new__(*args, **kwargs):   # real signature unknown
        """ 
        Create and return a new object.  See help(type) for accurate signature. 
        (创建并返回一个新对象)
        """
        pass

type的用法一:获取数据时那种类型 : type(object) -> the object’s type

a = 100
b = "100"
c = [11,22,33]
print(type(a))    # ======》 <class 'int'>
print(type(b))    # ======》 <class 'str'>
print(type(c))    # ======》 <class 'list'>

type的用法二:创建类:type(object_or_name, bases, dict)

1.在查看源码的时候,可以看到在初始化时,init__方法接受三个参数,type 在实例化的过程,也会重新创建一个新的类,而创建类的代码来自__new 方法,它的参数与 __init__方法是一样的。

2.当调用 type 进行实例化的时候,会先自动调用 new 方法,再调用__init__方法,最终会实例化一个对象,这个对象是一个类。

1. 元类 type 的 init 方法有3个参数:

 1.name:类名(字符串类型)
 2.bases:继承的父类(元祖类型)
 3.dict:属性和方法组成的字典(字典类型)

具体例子:

# 通过 class 定义类
class Myclass(object):
    a = 100
    b = 200
# 通过type创建的类(动态创建类)
Myclass1 = type("Myclass1",(object,),{"a":"100","b":"200"})
print(Myclass)
print(Myclass1)

如果需要定义实例方法和类属性怎么办呢?,将方法和属性以字典的形式传进去

def work(self):
    print("这是实例方法——————work————————")
# 定义类属性值
def init_method(self, aa, bb, cc):
    self.aa = aa
    self.bb = bb
    self.cc = cc
# 通过type创建的类(动态创建类)
Myclass2 = type("Myclass2",(object,),{"a":"100","b":"200","work":work,"work_1":work_1,"__init__":init_method})
m = Myclass2(11,22,33)
m.work()
print(m.aa, m.bb, m.cc)

四、自定义元类的使用

既然元类可以创建类,那也可以自定义元类,自定义直接继承类 type ,在自定义元类的步骤:

1.定义一个类继承type

2.重写new方法

具体例子:

# 定义一个类继承type
class Mytest(type):
	# 重写new方法
    def __new__(cls, type_name, bases, atter, *args, **kwargs):
        new_cls = super().__new__(cls,type_name, bases, atter)         
        return new_cls      # 返回新的类
M2 = Mytest("M2",(Mytest,),{"atter":100})
print(M2)     # =====》  <class '__main__.M2'>

使用class创建类的时候指定自定义的元类

1.不去指定时,默认创建都是type类

2.指定自定义的元类去创建类:metaclass = 指定的元类

class Myclass(type):
    """自定义的元类"""
    def __new__(cls, type_name, bases, attrs, *args, **kwargs):
        new_cls = super().__new__(cls, type_name, bases, attrs)
        print("这个是Myclass:", type_name, bases, attrs, )
        return new_cls
# 通过metaclass=xxxx   继承自定义元类
class Inherited_class(metaclass=Myclass):
    a = 100
    b = 200
print(type(Inherited_class))  # ======》 <class '__main__.Myclass'>

到此这篇关于Python超详细讲解元类的使用的文章就介绍到这了,更多相关Python元类内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Python做屏幕录制工具的实现示例

    Python做屏幕录制工具的实现示例

    本文主要介绍了Python做屏幕录制工具的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-06-06
  • 解读python基于netconf协议获取网元的数据

    解读python基于netconf协议获取网元的数据

    大多数企业都需要网络支撑企业的ICT运行,针对企业网络中的网元设备(包括交换机,路由器,防火墙等),很多企业希望根据自身的业务特点定制网络管理,如下就以华为的NE40E网元为例,说明如何通过python基于netconf协议实现对于网元配置数据的获取。
    2021-05-05
  • Python range函数生成一系列连续整数的内部机制解析

    Python range函数生成一系列连续整数的内部机制解析

    这篇文章主要为大家介绍了Python range函数生成一系列连续整数的内部机制解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • Python在日志中隐藏明文密码的方法

    Python在日志中隐藏明文密码的方法

    logging日志模块是python的一个内置模块,该模块定义了一些函数和类,为上层应用程序或库实现了一个强大而又灵活的日志记录系统,这篇文章主要介绍了Python如何在日志中隐藏明文密码 ,需要的朋友可以参考下
    2023-10-10
  • django template实现定义临时变量,自定义赋值、自增实例

    django template实现定义临时变量,自定义赋值、自增实例

    这篇文章主要介绍了django template实现定义临时变量,自定义赋值、自增实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-07-07
  • Django实现下载超大CSV文件的示例代码

    Django实现下载超大CSV文件的示例代码

    这篇文章主要为大家详细介绍了如何利用 Django 进行大型 CSV 文件的流传输,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2024-01-01
  • 基于django ManyToMany 使用的注意事项详解

    基于django ManyToMany 使用的注意事项详解

    今天小编就为大家分享一篇基于django ManyToMany 使用的注意事项详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-08-08
  • Python利用D3Blocks绘制可动态交互的图表

    Python利用D3Blocks绘制可动态交互的图表

    今天小编给大家来介绍一款十分好用的可视化模块,D3Blocks,不仅可以用来绘制可动态交互的图表,并且导出的图表可以是HTML格式,方便在浏览器上面呈现,感兴趣的可以了解一下
    2023-02-02
  • Python+Pygame实现彩色五子棋游戏

    Python+Pygame实现彩色五子棋游戏

    这篇文章主要为大家详细介绍了如何溧阳Python和Pygame实现彩色五子棋游戏,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-02-02
  • Python实现一个简单的MySQL类

    Python实现一个简单的MySQL类

    这篇文章主要介绍了Python实现一个简单的MySQL类,可实现基本的初始化连接及查询、删除等功能,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-01-01

最新评论