乾貨Python類和元類(metaclass)的理解和簡單運用

2021-03-02 數盟

 (一) python中的類

這裡討論的python類,都基於python2.7x以及繼承於object的新式類進行討論。

首先在python中,所有東西都是對象。這句話非常重要要理解元類我要重新來理解一下python中的類。

class Trick(object):

    pass

當python在執行帶class語句的時候,會初始化一個類對象放在內存裡面。例如這裡會初始化一個Trick對象。

這個對象(類)自身擁有創建對象(通常我們說的實例,但是在python中還是對象)的能力。

為了方便後續理解,我們可以先嘗試一下在新式類中最古老厲害的關鍵字type。

input:

class Trick(object):

pass

print type('123')

print type(123)

print type(Trick())

output:

<type 'str'>

<type 'int'>

<class '__main__.Trick'>

可以看到能得到我們平時使用的 str, int, 以及我們初始化的一個實例對象Trick()

但是下面的方法你可能沒有見過,type同樣可以用來動態創建一個類

type(類名, 父類的元組(針對繼承的情況,可以為空),包含屬性的字典(名稱和值))

這個怎麼用呢,我要用這個方法創建一個類 讓我們看下下面的代碼

input:

print type('trick', (), {})

output:

<class '__main__.trick'>

同樣我們可以實例化這個類對象

input:

print type('trick', (), {})()

output:

<__main__.trick object at 0x109283450>

可以看到,這裡就是一個trick的實例對象了。

同樣的這個方法還可以初始化創建類的父類,同時也可以初始化類屬性:

input:

class FlyToSky(object):

    pass

pw = type('Trick', (FlyToSky, ), {'laugh_at': 'hahahaha'})

print pw().laugh_at

print pw.__dict__

print pw.__bases__

print pw().__class__

print pw().__class__.__class__

output:

hahahaha

{'__module__': '__main__', 'laugh_at': 'hahahaha', '__doc__': None}

(<class '__main__.FlyToSky'>,)

<class '__main__.Trick'>

<type 'type'>

下面我將依次理一下上面的內容,在此之前我必須先介紹兩個魔法方法:

1、__class__這個方法用於查看對象屬於是哪個生成的,這樣理解在python中的所有東西都是對象,類對象也是對象。如果按照以前的思維來想的話就是類是元類的實例,而實例對象是類的實例。

2、__bases__這個方法用於得到一個對象的父類是誰,特別注意一下__base__返回單個父類,__bases__以tuple形式返回所有父類。

好了知道了這兩個方法我來依次說一下每行什麼意思。

1、使用type創建一個類賦值給pw type的接受的三個參數的意思分辨是(類的名稱, 類是否有父類(), 類的屬性字典{})

2、這裡初始化一個類的實例,然後嘗試去獲得父類的laugh_at屬性值,然後得到結果hahahaha

3、取一個pw的也就是我們常見類的類字典數據

4、拿到pw的父類,結果是我們指定的 FlyToSky

5、pw的實例pw()屬於哪個類初始化的,可以看到是class Trick

6、我們再看class trick是誰初始化的? 就是元類type了

(二) 什麼是元類以及簡單運用

這一切介紹完之後我們總算可以進入正題

到底什麼是元類?通俗的就是說,元類就是創建類的類。。。這樣聽起來是不是超級抽象?

來看看這個

Trick = MetaClass()

MyObject = Trick()

上面我們已經介紹了,搞一個Trick可以直接這樣

Trick = type('Trick', (), {})

可以這樣其實就是因為,Type實際上是一個元類,用他可以去創建類。什麼是元類剛才說了,元類就是創建類的類。也可以說他就是一個類的創建工廠。

類上面的__metaclass__屬性,相信願意了解元類細節的盆友,都肯定見過這個東西,而且為之好奇。不然我不知道是什麼支撐你看到這裡的。使用了__metaclass__這個魔法方法就意味著就會用__metaclass__指定的元類來創建類了。

class Trick(FlyToSky):

    pass

當我們在創建上面的類的時候,python做了如下的操作:

Trick中有__metaclass__這個屬性嗎?如果有,那麼Python會在內存中通過__metaclass__創建一個名字為Trick的類對象,也就是Trick這個東西。如果Python沒有找到__metaclass__,它會繼續在自己的父類FlyToSky中尋找__metaclass__屬性,並且嘗試以__metaclass__指定的方法創建一個Trick類對象。如果Python在任何一個父類中都找不到__metaclass__,它也不會就此放棄,而是去模塊中搜尋是否有__metaclass__的指定。如果還是找不到,好吧那就是使用默認的type來創建Trick。

那麼問題來了,我們要在__metaclass__中放置什麼呢?答案是可以創建一個類的東西,type,或者任何用到type或子類化type的東西都行。

(三) 自定義元類

自定義類的的目的,我總結了一下就是攔截類的創建,然後修改一些特性,然後返回該類。是不是有點熟悉?沒錯,就是感覺是裝飾器幹的事情,只是裝飾器是修飾一個函數,同樣是一個東西進去,然後被額外加了一些東西,最後被返回。

其實除了上面談到的制定一個__metaclass__並不需要賦值給它的不一定要是正式類,是一個函數也可以。要創建一個使所有模塊級別都是用這個元類創建類的話,在模塊級別設定__metaclass__就可以了。先寫一個來試試看,我還是延用stackoverflow上面那個哥們的例子,將所有的屬性都改為大寫的。

來看這個例子:

input:

def upper_attr(class_name, class_parents, class_attr):

    """

    返回一個對象,將屬性都改為大寫的形式

    :param class_name:  類的名稱

    :param class_parents: 類的父類tuple

    :param class_attr: 類的參數

    :return: 返回類

    """

    # 生成了一個generator

    attrs = ((name, value) for name, value in class_attr.items() if not name.startswith('__'))

    uppercase_attrs = dict((name.upper(), value) for name, value in attrs)

    return type(class_name, class_parents, uppercase_attrs)

__metaclass__ = upper_attr

pw = upper_attr('Trick', (), {'bar': 0})

print hasattr(pw, 'bar')

print hasattr(pw, 'BAR')

print pw.BAR

output:

False

True

0

可以從上面看到,我實現了一個元類(metaclass), 然後指定了模塊使用這個元類來創建類,所以當我下面使用type進行類創建的時候,可以發現小寫的bar參數被替換成了大寫的BAR參數,並且在最後我調用了這個類屬性並,列印了它。

上面我們使用了函數做元類傳遞給類,下面我們使用一個正式類來作為元類傳遞給__metaclass__

class UpperAttrMetaClass(type):

    def __new__(mcs, class_name, class_parents, class_attr):

        attrs = ((name, value) for name, value in class_attr.items() if not name.startswith('__'))

        uppercase_attrs = dict((name.upper(), value) for name, value in attrs)

        return super(UpperAttrMetaClass, mcs).__new__(mcs, class_name, class_parents, uppercase_attrs)

class Trick(object):

    __metaclass__ = UpperAttrMetaClass

    bar = 12

    money = 'unlimited'

print Trick.BAR

print Trick.MONEY

總結:

總之,就像我上面說的,略帶一點裝飾器的思路去理解元類這件事情,可能會讓你豁然開朗。元類這種黑暗魔法按照常理來說是不應該被廣泛使用的,從寫業務代碼一年差不多一年,除了在完成kepler項目的時候稍微黑魔法了一下(實際是根本不需要這樣操作),其他地方都沒有用到過。等到真正需要的時候,你可能不會去思考為什麼要去使用,而是因為要解決問題所以就是要這樣寫,所以才出現了元類這種東西。我是這樣理解的,一個東西存在的真正意義就在於你可以用這個東西去解決以前難以解決的問題,可以讓難以解決的問題變得簡單起來,而不是為了炫技讓一個問題變得複雜起來。


相關焦點

  • Python進階:一步步理解Python中的元類metaclass
    無論是數值、字符串、序列、字典、函數、模塊、類、實例、文件等等。元類(metaclass)是Python 2.2以後引入的概念,它的作用是定製類的創建行為。這麼解釋可能有點難理解,那麼這篇文章就通過實例,一步步解釋Python中的元類。
  • 【Python基礎】09-07 元類
    目錄一 元類介紹二 class關鍵字創建類的流程分析四 自定義元類控制類StanfordTeacher的創建五 自定義元類控制類StanfordTeacher的調用五 再看屬性查找六 作業                ✦ ✦ ✦ ✦ ✦ ✦一 元類介紹
  • 深入理解Python 類型和對象
    後面慢慢來揭曉. 豎線是來區別不同的區域的. 這個空白圖中, 我們來逐漸加上各種對象,畫上他們之間的關係,慢慢畫滿. 這個有助於幫助我們暫且擱置先入為主的各種有關面向對象的類和對象的觀念,以對象的方式(我們這裡的對象)來理解一切。 對象之間的關係我們使用兩種關係就可以連接我們所引入的這許多不同的對象.
  • 兩句話掌握Python最難知識點:元類
    學懂元類,你只需要知道兩句話:道生一,一生二,二生三,三生萬物我是誰?我從哪來裡?我要到哪裡去?在python世界,擁有一個永恆的道,那就是「type」,請記在腦海中,type就是道。如此廣袤無垠的python生態圈,都是由type產生出來的。
  • 夯實Java基礎系列9:深入理解Class類和Object類
    public class 多線程中的回調 { //這裡簡單地使用future和callable實現了線程執行完後 public static void main(String[] args) throws ExecutionException, InterruptedException {
  • 詳解Python中的__init__和__new__的區別
    當使用 Persion(name, age) 這樣的表達式來實例化一個類時,最先被調用的方法 其實是 __new__ 方法。二、__new__ 方法是什麼?__new__方法接受的參數雖然也是和__init__一樣,但__init__是在類實例創建之後調用,而 __new__方法正是創建這個類實例的方法。
  • 詳解Python元類
    理解元類(metaclass)之前,我們先了解下Python中的OOP和類(Class)。面向對象全稱 Object Oriented Programming 簡稱OOP,這種編程思想被大家所熟知。它是把對象作為一個程序的基本單元,把數據和功能封裝在裡面,能夠實現很好的復用性,靈活性和擴展性。
  • 全面深入理解 Python 類與對象
    -中.html類屬性和實例屬性查找順序屬性:在內部定義的方法或者變量使用代碼:class magic: a = 'langzi' def __init__(self,x): self.x = x def run(self): return
  • Python學習:類和實例
    ,這個思想就我個人的體會,感覺很重要,除了封裝的功能外,類作為一種規範,我們自己可以定製的規範,從這個角度來看,在以後我們學習設計模式的時候,對設計模式的理解會很有幫助。它屬於類,和實例無關。建議只使用類名.靜態方法的調用方式。
  • python 類的繼承和派生
    什麼是繼承繼承是一種創建類的方法,在python中,一個類可以繼承來自一個或多個父類。原始類稱為基類或超類。__bases__(<class '__main__.ParentClass1'>, <class '__main__.ParentClass2'>)什麼時候用繼承假如已經有幾個類,而類與類之間有共同的變量屬性和函數屬性,那就可以把這幾個變量屬性和函數屬性提取出來作為基類的屬性。
  • [每日一題]2、Python中的類的定義和裝飾器@classmethod與@staticmethod
    最近創建了一個GitHub項目,主要會分享一些Python方面面試題和練習題,訪問地址:https://github.com/python3xxx/Python-Every-Day(點擊閱讀原文,直達)Python Every Day, 第 2 天python中的定義類方法有三種形式
  • Python-15-類的定義和使用
    官方定義類提供了一種組合數據和功能的方法每個類的實例可以擁有保存自己狀態的屬性。一個類的實例也可以有改變自己狀態的(定義在類中的)方法。簡單的說,類就是一個事物的抽象描述。所以類中可以包含描述類的方法和屬性,其中方法又分為普通方法,類方法,靜態方法。詳細區別可查看Python的@staticmethod和@classmethod的作用與區別,今天主要介紹普通方法。
  • 【Python面試】談談對 Python3 和 Python2 的區別?​
    如果參考答案不夠好,或者有錯誤的話,麻煩大家可以在留言區給出自己的意見和討論,大家是要一起學習的 。廢話不多說,開始今天的題目:問:談談Python3 和 Python2 的區別?答:Python3跟Python2比,語法上就有很多區別,都需要特別注意,下面給大家列舉幾個常見的 。
  • python:創建類和根據類創建實例
    1.餐館:創建一個名為Restaurant的類,其方法_init_()設置兩個屬性:restaurant_name和cuisine_type.創建一個名為describe_restaurant()的方法和一個名為open_restaurant()的方法,其中前者列印前述兩項信息
  • 如何理解python中的類和對象?
    什麼是類和對象類和對象,在我們的生活中其實是很容易找例子的。類是一種把對象分組歸類的方法。比如動物,植物就可以看作是類,而大象,獅子就可以看作一個動物類中的對象;花,草可以看作是植物類中的對象。為什麼大象和獅子就劃分為動物類,花和草就劃分為植物類呢?
  • 從0開始學python第8.1節-類和對象
    我們今天學習的面向對象編程則將程序做了更高維度的封裝,用人類更容易理解的方式編程。類&實例中國道德經有一句話:道生一,一生二,二生三,三生萬物,講的是世間的萬物都是從特定的模板裡生出來的。這個模板就是類,生出來的東西就是對象。我們來看幾個案例:寵物狗有很多種:泰迪、金毛、哈士奇 都屬於狗科動物。
  • Python 中 staticmethod 和 classmethod 原理探究
    如果能理解那邊描述符的使用方式,那也能很快理解本篇中的 staticmethod 和 classmethod 。這麼簡單的代碼也已經是 C 實現版本對應的Python完整代碼了。classmethod 的實現classmethod 則是要讓 C.f和 c.f 都返回方法,並且傳遞隱式參數 cls , 運行代碼如下:class C:    @classmethod
  • 乾貨!Python入門基礎之面向對象二:類和實例、方法
    前面一篇文章我介紹了python面向對象的基本知識,連結在最下面。初步解釋了面向對象和面向對象的優點,今天就開始正式用代碼來展現面向對象。1、利用class創建類Python中,創建類的語句是如下所示先解釋一下,class後面跟的是類名,括號裡面是基類(也成為父類)python3中默認繼承object。
  • Python中你不知道的事:多繼承和類的方法,以及什麼是類的特殊方法
    可以同時繼承多個父類Python中可以有兩個以上的父類,當有A,B,C三個父類的時候,第三個父類可以同時繼承A與B兩類,C可以使用A與B中的屬性和方法。輸出結果python 給繼承關係排列了一個順序,我們將它稱之為繼承樹。用print(D.
  • Python每天一分鐘:類定義進階/炫技—使用type函數動態創建類
    Python中的type函數是常用於查看變量類型,在調試python代碼和bug修復過程中都是非常有效的工具。該怎麼理解呢?實際上從Python解釋器的角度就能說得通:python在使用關鍵字class 定義 testClass類時,可理解為創建了一個特殊的對象(type類的對象)且將該對象賦給了testClass變量。