Python 元編程

来源:https://www.cnblogs.com/rockety/archive/2023/04/08/17298553.html
-Advertisement-
Play Games

作者:袁首京 原創文章,轉載時請保留此聲明,並給出原文連接。 元編程並不象它聽起來那麼時髦和新奇。常用的 decorator 就可以認為是一種元編程。簡單來說,元編程就是編寫操作代碼的代碼。 有點繞,是吧?彆著急,咱們一點一點來討論。 註意:本文中的代碼適用於 Python 3.3 及以上。 元類 ...


作者:袁首京

原創文章,轉載時請保留此聲明,並給出原文連接。

元編程並不象它聽起來那麼時髦和新奇。常用的 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,可能會用到;
作者:袁首京

原創文章,轉載時請保留此聲明,並給出原文連接。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 今年是23年,互聯網大裁員,電腦行業的小伙伴也深有體會,那麼還沒有入行的我們要怎麼去選擇編程語言?一文簡單帶你分析你應該值得去學什麼 原文地址,未來會持續更新Python面試題、前後端分離項目,點擊鏈接前往 結論 值得去學Python,不管是作為第一編程語言還是第二編程語言,你都應該要學習Pyth ...
  • 效果 搭建一個spring源碼調試環境,創建一個spring-demo模塊,寫一些測試代碼。 給源碼添加註釋。 給源碼打包 ubantu環境下搭建spring6.0.x源碼環境 步驟 源碼網址 Spring Framework 下載代碼 fork到自己的GitHub倉庫,然後拉代碼 git clon ...
  • 簡介 本文給大家推薦博主自己開源的電商項目newbee-mall-pro。在newbee-mall項目的基礎上搭建而來, 使用 mybatis-plus 作為 orm 層框架,並添加了一系列高級功能以及代碼優化,特性如下: 商城首頁 【為你推薦】 欄目添加協同過濾演算法。按照 UserCF 基於用戶的 ...
  • 操作系統 :CentOS 7.6_x64 freeswitch版本 :1.10.9 sofia-sip版本: sofia-sip-1.13.14 freeswitch使用sip協議進行通信,當sip消息超過mtu時,會出現拆包的情況,這裡整理下sip消息拆包原理及組包流程。 一、拆包的原理 簡單來說 ...
  • 函數式語言特性:-迭代器和閉包 本章內容 閉包(closures) 迭代器(iterators) 優化改善 12 章的實例項目 討論閉包和迭代器的運行時性能 一、閉包(1)- 使用閉包創建抽象行為 什麼是閉包(closure) 閉包:可以捕獲其所在環境的匿名函數。 閉包: 是匿名函數 保存為變數、作 ...
  • 承接上文 承接上一篇文章【演算法數據結構專題】「延時隊列演算法」史上手把手教你針對層級時間輪(TimingWheel)實現延時隊列的開發實戰落地(上)】我們基本上對層級時間輪演算法的基本原理有了一定的認識,本章節就從落地的角度進行分析和介紹如何通過Java進行實現一個屬於我們自己的時間輪服務組件,最後,在 ...
  • 實現一個簡單的UDP通信程式,僅作為筆記使用 網路編程中有三要素:IP、埠號和通信協議,分別用來確定對方在互聯網上的地址、指定接受數據的軟體和確定數據在網路中傳輸的規則。 IP地址 IP地址分為IPv4地址和IPv6地址,這裡不做討論。 IPv4地址中分為公網地址(萬維網使用)和私有地址(區域網使 ...
  • 第二章 線程管控 std::thread 簡介 構造和析構函數 /// 預設構造 /// 創建一個線程,什麼也不做 thread() noexcept; /// 帶參構造 /// 創建一個線程,以 A 為參數執行 F 函數 template <class Fn, class... Args> exp ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...