Python 元编程

yizhihongxing
作者:袁首京

原创文章,转载时请保留此声明,并给出原文连接。

元编程并不象它听起来那么时髦和新奇。常用的 decorator 就可以认为是一种元编程。简单来说,元编程就是编写操作代码的代码。

有点绕,是吧?别着急,咱们一点一点来讨论。

注意:本文中的代码适用于 Python 3.3 及以上。

元类

多数编程语言中,一切东西都有类型。Python 也不例外,我们可以用 type() 函数获取任意变量的类型。

num = 23
print("Type of num is:", type(num))

lst = [1, 2, 4]
print("Type of lst is:", type(lst))

name = "Atul"
print("Type of name is:", type(name))

执行结果是:

Type of num is: <class 'int'>
Type of lst is: <class 'list'>
Type of name is: <class 'str'>

Python 中的所有类型都是由 Class 定义的。这一条与其它编程语言,比如 Java、C++ 等等不同。在那些语言中,int、char、float 之类是基本数据类型,但是在 Python 中,它们是 int 类或 str 类的对象。

象其它 OOP 语言一样,我们可以使用 class 定义新类型:

class Student:
    pass

stu_obj = Student()
print("Type of stu_obj is:", type(stu_obj))

执行结果是:

Type of stu_obj is: <class '**main**.Student'>

一点儿也不意外,对吧?其实有意外,因为在 Python 中,类也是一个对象,就像任何其他对象一样,它是元类的实例。即一个特殊的类,创建了 Class 这个特殊的类实例。看如下代码:

class Student:
    pass

print("Type of Student class is:", type(Student))

执行结果是:

Type of Student class is: <class 'type'>

既然类也是一个对象,所以以修改对象相同的方式修改它就顺理成章。如下先定义一个没有任何属性和方法的类,然后在外部为其添加属性和方法:

class test:
    pass

test.x = 45
test.foo = lambda self: print('Hello')

myobj = test()
print(myobj.x)
myobj.foo()

执行结果是:

45
Hello

以上过程可以简单概括为:

元类创建类,类创建实例

画个图象这样:

元类 -> 类 -> 实例

因此,我们就可以编写自定义的元类,执行额外的操作或者注入代码,来改变类的生成过程。这在某些场景下很有用,主要是比如有些情况下使用元编程更简单,另一些情况只有元编程才能解决问题。

创建自定义元类

创建自定义元类,有两种方法。第一种是继承 type 元类,并且覆写两个方法:

  1. new()

它在 init() 之前调用,生成类实例并返回。我们可以覆盖此方法来控制对象的创建过程。

  1. init()

这个不多解释,相信你都明白。

如下是个例子:

class MultiBases(type):
    def __new__(cls, clsname, bases, clsdict):
        if len(bases)>1:
            raise TypeError("Inherited multiple base classes!!!")

        return super().__new__(cls, clsname, bases, clsdict)


class Base(metaclass=MultiBases):
    pass


class A(Base):
    pass


class B(Base):
    pass


class C(A, B):
    pass

执行结果是:

Traceback (most recent call last):
File "<stdin>", line 2, in <module>
File "<stdin>", line 8, in **new**
TypeError: Inherited multiple base classes!!!

第二种方法是直接使用 type() 函数创建类。这个方法如果只用一个参数调用,它会返回该参数的类型,前文已经描述过。但是使用三个参数调用时,它会创建一个类。这三个参数如下:

  1. 类名称;
  2. 继承的父类的元组。你没看错,是元组,别忘了 Python 可以多继承;
  3. 一个字典。定义类属性和方法;

以下是示例:

def test_method(self):
    print("This is Test class method!")


class Base:

    def myfun(self):
        print("This is inherited method!")


Test = type('Test', (Base, ), dict(x="atul", my_method=test_method))
print("Type of Test class: ", type(Test))

test_obj = Test()
print("Type of test_obj: ", type(test_obj))

test_obj.myfun()
test_obj.my_method()

print(test_obj.x)

执行结果是:

Type of Test class: <class 'type'>
Type of test_obj: <class '**main**.Test'>
This is inherited method!
This is Test class method!
atul

使用元类解决问题

了解了元类的创建方法后,可以来解决一些实际问题了。例如,如果我们想在每次调用类方法时,都先输出一下它的全限定名,该怎么办呢?

最常用的方法是使用 decorator,象这样:

from functools import wraps


def debug(func):

    @wraps(func)
    def wrapper(*args, **kwargs):
        print("Full name of this method:", func.__qualname__)
        return func(*args, **kwargs)
    return wrapper


def debugmethods(cls):
    for key, val in vars(cls).items():
        if callable(val):
            setattr(cls, key, debug(val))
    return cls


@debugmethods
class Calc:

    def add(self, x, y):
        return x+y

    def mul(self, x, y):
        return x\*y

    def div(self, x, y):
        return x/y


mycal = Calc()
print(mycal.add(2, 3))
print(mycal.mul(5, 2))

执行结果是:

Full name of this method: Calc.add
5
Full name of this method: Calc.mul
10

这个方案很漂亮。但是,如果变更一下需求,例如我们希望 Calc 的所有子类的方法执行时,都先输出一下它的全限定名,该怎么办呢?

在每一个子类上加上 @debugmethods 是一种方案,但是有点啰嗦,是不是?

该基于元类的解决方案出场了,以下是个例子:

from functools import wraps


def debug(func):

    @wraps(func)
    def wrapper(*args, **kwargs):
        print("Full name of this method:", func.__qualname__)
        return func(*args, **kwargs)
    return wrapper


def debugmethods(cls):

    for key, val in vars(cls).items():
        if callable(val):
            setattr(cls, key, debug(val))
    return cls


class debugMeta(type):

    def __new__(cls, clsname, bases, clsdict):
        obj = super().__new__(cls, clsname, bases, clsdict)
        obj = debugmethods(obj)
        return obj


class Base(metaclass=debugMeta):
    pass


class Calc(Base):

    def add(self, x, y):
        return x+y


class Calc_adv(Calc):

    def mul(self, x, y):
        return x\*y


mycal = Calc_adv()
print(mycal.mul(2, 3))

执行结果是:

Full name of this method: Calc_adv.mul
6

何时使用元类

该说的基本说完了,剩下最好一件事。元编程算是 Python 的一个魔法,多数时候我们其实用不到。但是什么时候需要呢?大概有三种情况:

  • 如果我们想要一个特性,沿着继承层次结构向下传递,可以用;
  • 如果我们想在类创建后,能动态修改,可以用;
  • 如果我们是在开发类库或者 API,可能会用到;
作者:袁首京

原创文章,转载时请保留此声明,并给出原文连接。

原文链接:https://www.cnblogs.com/rockety/p/17298553.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Python 元编程 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • 将string类型的数据类型转换为spark rdd时报错的解决方法

    当我们将string类型的数据转换为Spark RDD时,有时会遇到以下常见的报错信息:“TypeError: Can not infer schema for type: ”。这是因为Spark无法推断出string类型的数据的schema,需要我们手动指定schema。 以下是将string类型的数据转换为Spark RDD的解决方法: 手动指定sche…

    python 2023年6月6日
    00
  • python import模块时有错误红线的原因

    当我们在Python中导入模块时,有时会出现错误红线,这通常是由于以下原因之一引起的: 模块不存在或路径不正确 模块中存在语法错误。 模块中存在行时错误。 以下是解决这些问题方法: 模块不存在或路径不正确 当我们导入模块时,如果模块不存在或路径不正确,就会出现错误红线。是解决这个问题的方法: 检查模块存在。 检查模块路径是否正确。 例如,我们可以使用以下代码…

    python 2023年5月13日
    00
  • Python 中的函数装饰器和闭包详解

    Python中的函数装饰器和闭包都是高级的语言特性,熟练掌握这些特性可以提高代码的可复用性和可读性。本文将分为以下几部分对函数装饰器和闭包进行详细讲解: 函数装饰器 一个函数装饰器是一个可以接受一个函数作为输入并返回一个新函数的可调用对象。使用装饰器可以在不修改原函数的情况下,将新的行为附加到函数上。这种技术被称为元编程。 函数装饰器是 Python 中最常…

    python 2023年6月3日
    00
  • python Tkinter是什么

    Python Tkinter是一个Python标准库,用于构建GUI应用程序的工具包。Tkinter提供了内置的GUI组件,如按钮、标签、文本框和滚动条,有助于创建互动和易于使用的Python应用程序。 一些Tkinter的特点如下: 可以在各种操作系统中使用,包括Windows、macOS和Linux等。 Tkinter接口具有很多功能,可以创建可扩展的G…

    python 2023年6月13日
    00
  • python字符串与url编码的转换实例

    Python字符串与URL编码的转换是编程时常见的操作之一。在这里我将分享一下如何在Python中进行字符串和URL编码之间的转换。 Python字符串与URL编码的转换实例 对于Python字符串和URL编码之间的转换,我们可以使用urllib库。urllib是Python的一个标准库,它提供了处理URL的各种方法。 1. 将字符串进行URL编码 在Pyt…

    python 2023年5月31日
    00
  • Python3使用pandas模块读写excel操作示例

    下面是Python3使用pandas模块读写Excel操作示例的完整实例教程。 前置条件 在开始本教程前,请确保你已经安装好了以下的环境: Python3 pandas xlrd(用于读取Excel文件) openpyxl(用于写入Excel文件) 如果你还没有安装,请先安装上述环境。 读取Excel文件 在Python中,我们可以使用pandas模块来读取…

    python 2023年5月13日
    00
  • python爬虫学习笔记之pyquery模块基本用法详解

    Python爬虫是一种常见的网络爬虫技术,可以用于从网站上获取数据。PyQuery是一个Python库,它提供了类似于jQuery的语法,可以方便地解析HTML和XML文档。以下是Python爬虫学习笔记之PyQuery模块基本用法详解,包含两个示例。 示例1:解析HTML文档 以下是一个示例,可以使用PyQuery解析HTML文档: from pyquer…

    python 2023年5月15日
    00
  • python数据解析之XPath详解

    XPath是一种用于在XML文档中定位元素和属性的语言。Python提供了多种解析XML数据的方法,其中包括使用XPath表达式解析XML数据。以下是详细讲解Python数据解析之XPath详解,包含两个示例。 示例1:使用XPath解析XML 以下是一个示例,可以使用XPath解析XML: from lxml import etree # 定义XML文档 …

    python 2023年5月15日
    00
合作推广
合作推广
分享本页
返回顶部