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

Stella981
• 阅读 342

摘要:通过本文,将深入讨论Python元类,其属性,如何以及何时在Python中使用元类。

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

    • 什么是Python元类?
    • Python中的类和对象
    • Python中的动态类
    • Python元类如何工作?
      • 类型类
      • Python中的自定义元类
    • 装饰器vs元类

什么是Python元类?

Python元类是与Python的面向对象编程概念相关的高级功能之一。它确定类的行为,并进一步帮助其修改。

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

用Python创建的每个类都有一个基础的Metaclass。因此,在创建类时,您将间接使用元类。它隐式发生,您无需指定任何内容。

与元编程相关联的元类决定了程序对其自身进行操作的能力。 学习元类可能看起来很复杂,但是让我们先从一些类和对象的概念入手,以便于理解。

Python中的类和对象

类是一个蓝图,是具有对象的逻辑实体。 一个简单的类在声明时没有分配任何内存,它是在创建一个类的实例时发生的。

通过创建的对象,可以访问该类。该类仅用作模板。对象的属性本质上意味着我们可以在运行时与它进行交互,传递诸如变量之类的参数,进行存储,修改,也可以与它进行交互。

可以使用__class__属性检查对象的类。让我们看一个简单的例子:

class Demo: 
       pass       
#This is a class named demo
 test=Demo()
print(test.__class__)  #shows class of obj
print(type(test))  #alternate method

Output: <class '__main__.Demo'>

Python大量处理类和对象的概念,并允许轻松,顺利地进行应用程序开发。但是,什么使Python与Java和C这样的语言不同呢?**Python中的所有内容都可以定义为具有属性和方法的对象。** 主题演讲是Python中的类不过是更大类的另一个对象。

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

类为对象定义规则。同样,元类负责为类分配行为。我们已经知道,类是对象,就像每个对象都有一个实例一样,类是元类的实例。

但是也有像Ruby和Objective-C这样的语言也支持元类。那么,是什么使Python Metaclass更好,为什么还要学习它呢?答案是Python中的动态类。让我们仔细看看。

Python中的动态类

Python是一种动态编程语言,并允许在运行时创建类。与C ++等其他语言不同,后者仅允许在编译时创建类。在灵活性方面,Python优于其他静态类型的语言。

动态和静态类型语言之间的差异并不大, 但是在Python中,它由于提供元编程而变得更加有用。

但是,如果我告诉您还有另一个关键功能将Python与其他编程语言区分开呢?

诸如Java或C ++之类的语言具有float,char,int等数据类型,而Python将每个变量视为对象。每个对象都属于一个类,例如int类或str类。您可以使用称为type()的内置函数来简单地检查任何变量的类

number = 10993
print("Type associated is:", type(number))
name = "Aishwarya"
print("Type associated is:", type(name))

Output:

Type associated is: <class 'int'>

Type associated is: <class 'str'>

现在,您了解了Python中的所有内容都有与之关联的类型。在下一个主题中,我们将尝试了解元类实际上是如何工作的。

Python元类如何工作?

每当创建一个类时,都会调用默认的Metaclass类型。 元类包含名称,基类集以及与该类关联的属性等信息。因此,在实例化一个类时,将调用带有这些参数的类。可以通过两种方法创建元类:

  1. 类型类
  2. 自定义元类

让我们继续输入class以及如何创建class。

类型类

Python有一个称为type的内置元类。与Java或C不同,那里有主要的数据类型。Python中的每个变量或对象都有一个与之关联的类。Python使用幕后的Type类创建所有类。在上一个主题中,我们看到了如何使用type()检查对象的类。让我们举一个例子,说明如何通过创建一个简单的类来定义新类型。

class Edureka():
obj = Edureka()
 
print(type(obj))

Output: <class '__main__.Edureka'>

print(type(Edureka))

Output: <class 'type'>

在上面的代码中,我们有一个名为Edureka的类,以及一个关联的对象。我们通过简单地在该类型之后创建一个名为自身的类,创建了一个名为Edureka的新类型。在第二个代码中,当我们检查Edureka类的类型时,其结果为“类型”。

因此,除非另有定义,否则元类使用类型类来创建所有其他类。我们可以通过两种方法访问Type类:

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

当我们通过类型类传递参数时,它使用以下语法。

type(__name__, __base__, attributes)
  • 名称是一个字符串,并带有类名
  • 该基础是一个元组,可帮助创建子类
  • 属性是字典,并分配键值对

由于Python中的类的行为与对象相似,因此可以用相同的方式更改其行为。我们可以在类内添加或删除方法,类似于对对象的处理方式。

现在您已经知道Metaclass在Python中创建了所有其他类,并使用类型class定义了它们的行为。但是,您一定想知道,我们还有其他方法可以创建元类吗?因此,让我们看看如何创建一个自定义的元类。

Python中的自定义元类

现在我们知道并理解类型类如何工作。现在该学习如何创建自定义元类了。我们可以通过执行动作或代码注入来修改类的工作。为此,我们可以在创建类定义时将Metaclass作为关键字传递。另外,我们可以通过简单地继承通过此Metaclass关键字实例化的类来实现此目的。

在创建新类时,Python查找__metaclass__ 关键字。以防万一,如果不存在。它遵循类型类层次结构。

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

Python在命名空间中执行所有字典后,将调用类型对象,后者创建类的对象。我们可以使用两种方法来创建自定义元类。

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

class EduFirst(type):
def __new__(cls, name, base_cls, dict):
pass
class EduSecond(type):
def __init__(self, name, base_cls, dict):
pass

让我详细解释这两种方法:

  1. __new __(): 当用户要在类创建之前定义元组字典时使用。它返回一个类的实例,并且很容易覆盖/管理对象流。
  2. __init __():在创建对象并对其进行初始化之后调用它。

Python中的__call__是什么?

在正式的Python文档中,**__call__**方法可用于定义自定义元类。同样,当调用类定义自定义行为时,我们可以覆盖__prepare__之类的其他方法。

就像类如何像创建对象的模板一样,元类也像类创建模板一样。因此,元类也称为类工厂。

请参见下一个示例:

class Meta(type):
def __init__(cls, name, base, dct):
cls.attribute = 200
class Test(metaclass = Meta):
pass
Test.attribute

Output: 200

元类允许自定义类。还有多种其他有效且简单得多的方法可以通过这些方法实现相同的输出。这样的例子之一就是使用装饰器。

装饰器vs元类

Decorator是Python的一项流行功能,它允许您向代码中添加更多功能。装饰器是可调用的对象,可帮助修改现有的类甚至函数。在编译期间,部分代码将调用并修改另一部分。此过程也称为元编程。

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

def decorator(cls):
class NewClass(cls):
attribute = 200
 return NewClass
@decorator
Class Test1:
 pass
@decorator
 
Class Test2:
 pass
Test1.attribute
 
Test2.attribute

Output: 200

Python中的Decorator是一个非常有用且功能强大的工具,可帮助您更改函数的行为,而无需实际更改任何代码。 当您要在调试时修改程序的一部分而不是重写函数或更改整个程序时,这非常方便。取而代之的是,您只需编写一个单行装饰器,其余的就由它来处理。

本文分享自华为云社区《如何创建您的第一个Python元类?》,原文作者:Yuchuan。

点击关注,第一时间了解华为云新鲜技术~

点赞
收藏
评论区
推荐文章
blmius blmius
1年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录 问题 用navicat导入数据时,报错: 原因这是因为当前的MySQL不支持datetime为0的情况。 解决修改sql\mode: sql\mode:SQL Mode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。 全局s
Karen110 Karen110
1年前
​一篇文章总结一下Python库中关于时间的常见操作
前言本次来总结一下关于Python时间的相关操作,有一个有趣的问题。如果你的业务用不到时间相关的操作,你的业务基本上会一直用不到。但是如果你的业务一旦用到了时间操作,你就会发现,淦,到处都是时间操作。。。所以思来想去,还是总结一下吧,本次会采用类型注解方式。 time包import time 时间戳 从1970年1月1日00:00:00标准时区诞生到现在
Wesley13 Wesley13
11个月前
thinkphp3.2.3模板渲染支持三元表达式
thinkphp3.2.3模板渲染支持三元表达式 {$status?'正常':'错误'} {$info['status']?$info['msg']:$info['error']} 注意:三元运算符中暂时不支持点语法。 如下:            <div class="modal hide fade" id='myModa
Stella981 Stella981
11个月前
KVM调整cpu和内存
一.修改kvm虚拟机的配置 1、virsh edit centos7 找到“memory”和“vcpu”标签,将 <name>centos7</name> <uuid>2220a6d1-a36a-4fbb-8523-e078b3dfe795</uuid>
Stella981 Stella981
11个月前
Python之time模块的时间戳、时间字符串格式化与转换
Python处理时间和时间戳的内置模块就有`time`,和`datetime`两个,本文先说`time`模块。 ### 关于时间戳的几个概念 * 时间戳,根据1970年1月1日00:00:00开始按秒计算的偏移量。 * 时间元组(`struct_time`),包含9个元素。  `time.struct_time(tm_y
Wesley13 Wesley13
11个月前
java常用类(2)
三、时间处理相关类 Date类:计算机世界把1970年1月1号定为基准时间,每个度量单位是毫秒(1秒的千分之一),用long类型的变量表示时间。 Date分配Date对象并初始化对象,以表示自从标准基准时间(称为“历元”(epoch),即1970年1月1日08:00:00GMT)以来的指定毫秒数。 示例: package cn.tanjian
Stella981 Stella981
11个月前
ASMSupport教程4.9 生成三元运算符
<p>这节我们介绍如何用ASMSupport生成三元运算符(... ? ... : ... )运算符。我们预计生成如下代码:</p> <div id="scid:9D7513F9-C04C-4721-824A-2B34F0212519:935e30cc-3321-4e00-93ba-9834f3a4e044" class="wlWriterEditableS
Wesley13 Wesley13
11个月前
MySQL查询按照指定规则排序
1.按照指定(单个)字段排序 select * from table_name order id desc; 2.按照指定(多个)字段排序 select * from table_name order id desc,status desc; 3.按照指定字段和规则排序 selec
Wesley13 Wesley13
11个月前
PHP中的NOW()函数
是否有一个PHP函数以与MySQL函数`NOW()`相同的格式返回日期和时间? 我知道如何使用`date()`做到这一点,但是我问是否有一个仅用于此的函数。 例如,返回: 2009-12-01 00:00:00 * * * ### #1楼 使用此功能: function getDatetimeNow() {
Wesley13 Wesley13
11个月前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
#### 背景描述 # Time: 2019-01-24T00:08:14.705724+08:00 # User@Host: **[**] @ [**] Id: ** # Schema: sentrymeta Last_errno: 0 Killed: 0 # Query_time: 0.315758 Lock_
helloworld_34035044 helloworld_34035044
2个月前
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。 uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid() 或 uuid(sep)参数说明:sep 布尔值,生成的uuid中是否包含分隔符'',缺省为