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

Stella981
• 阅读 729

摘要:通过本文,将深入讨论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
3年前
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:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Wesley13 Wesley13
3年前
java常用类(2)
三、时间处理相关类Date类:计算机世界把1970年1月1号定为基准时间,每个度量单位是毫秒(1秒的千分之一),用long类型的变量表示时间。Date分配Date对象并初始化对象,以表示自从标准基准时间(称为“历元”(epoch),即1970年1月1日08:00:00GMT)以来的指定毫秒数。示例:packagecn.tanjian
Java修道之路,问鼎巅峰,我辈代码修仙法力齐天
<center<fontcolor00FF7Fsize5face"黑体"代码尽头谁为峰,一见秃头道成空。</font<center<fontcolor00FF00size5face"黑体"编程修真路破折,一步一劫渡飞升。</font众所周知,编程修真有八大境界:1.Javase练气筑基2.数据库结丹3.web前端元婴4.Jav
Stella981 Stella981
3年前
Python3:sqlalchemy对mysql数据库操作,非sql语句
Python3:sqlalchemy对mysql数据库操作,非sql语句python3authorlizmdatetime2018020110:00:00coding:utf8'''
Wesley13 Wesley13
3年前
thinkphp3.2.3模板渲染支持三元表达式
thinkphp3.2.3模板渲染支持三元表达式{$status?'正常':'错误'}{$info'status'?$info'msg':$info'error'}注意:三元运算符中暂时不支持点语法。如下:           <divclass"modalhidefade"id'myModa
Stella981 Stella981
3年前
Python 学生管理
原文链接: Python学生管理(https://my.oschina.net/ahaoboy/blog/1526102)python3练手 codingutf8学生类classStu:三个字段id字符串,name字符串,score小数def__init__(sel
Stella981 Stella981
3年前
Python之time模块的时间戳、时间字符串格式化与转换
Python处理时间和时间戳的内置模块就有time,和datetime两个,本文先说time模块。关于时间戳的几个概念时间戳,根据1970年1月1日00:00:00开始按秒计算的偏移量。时间元组(struct_time),包含9个元素。 time.struct_time(tm_y
Python进阶者 Python进阶者
10个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这