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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...