成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

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

開發 前端
這里討論的python類,都基于python2.7x以及繼承于object的新式類進行討論。首先在python中,所有東西都是對象。這句話非常重要要理解元類我要重新來理解一下python中的類。

 (一) python中的類

今天看到一篇好文,然后結合自己的情況總結一波。

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

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

  1. class Trick(object): 
  2.     pass 

 

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

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

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

  1. input: 
  2. class Trick(object): 
  3. pass 
  4.  
  5. print type('123'
  6. print type(123) 
  7. print type(Trick()) 
  8.  
  9. output
  10. <type 'str'
  11. <type 'int'
  12. <class '__main__.Trick' 

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

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

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

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

  1. input: 
  2. print type('trick', (), {}) 
  3.  
  4. output
  5. <class '__main__.trick'
  6. 同樣我們可以實例化這個類對象 
  7.  
  8. input: 
  9. print type('trick', (), {})() 
  10.  
  11. output
  12. <__main__.trick object at 0x109283450>  

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

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

  1. input: 
  2. class FlyToSky(object): 
  3.     pass 
  4.  
  5. pw = type('Trick', (FlyToSky, ), {'laugh_at''hahahaha'}) 
  6. print pw().laugh_at 
  7. print pw.__dict__ 
  8. print pw.__bases__ 
  9. print pw().__class__ 
  10. print pw().__class__.__class__ 
  11.  
  12.  
  13. output
  14. hahahaha 
  15. {'__module__''__main__''laugh_at''hahahaha''__doc__': None} 
  16. (<class '__main__.FlyToSky'>,) 
  17. <class '__main__.Trick'
  18. <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了

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

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

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

來看看這個

  1. Trick = MetaClass() 
  2. MyObject = Trick()  

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

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

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

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

  1. class Trick(FlyToSky): 
  2.     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上面那個哥們的例子,將所有的屬性都改為大寫的。🤗

來看這個例子:

  1. input: 
  2. def upper_attr(class_name, class_parents, class_attr): 
  3.     ""
  4.     返回一個對象,將屬性都改為大寫的形式 
  5.     :param class_name:  類的名稱 
  6.     :param class_parents: 類的父類tuple 
  7.     :param class_attr: 類的參數 
  8.     :return: 返回類 
  9.     ""
  10.     # 生成了一個generator 
  11.     attrs = ((name, value) for name, value in class_attr.items() if not name.startswith('__')) 
  12.     uppercase_attrs = dict((name.upper(), value) for name, value in attrs) 
  13.     return type(class_name, class_parents, uppercase_attrs) 
  14.  
  15. __metaclass__ = upper_attr 
  16.  
  17. pw = upper_attr('Trick', (), {'bar': 0}) 
  18. print hasattr(pw, 'bar'
  19. print hasattr(pw, 'BAR'
  20. print pw.BAR 
  21.  
  22. output
  23. False 
  24. True 
  25.  

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

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

  1. class UpperAttrMetaClass(type): 
  2.     def __new__(mcs, class_name, class_parents, class_attr): 
  3.         attrs = ((name, value) for name, value in class_attr.items() if not name.startswith('__')) 
  4.         uppercase_attrs = dict((name.upper(), value) for name, value in attrs) 
  5.         return super(UpperAttrMetaClass, mcs).__new__(mcs, class_name, class_parents, uppercase_attrs) 
  6.  
  7.  
  8. class Trick(object): 
  9.     __metaclass__ = UpperAttrMetaClass 
  10.     bar = 12 
  11.     money = 'unlimited' 
  12.  
  13. print Trick.BAR 
  14. print Trick.MONEY  

總結:

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

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2016-10-25 14:27:32

metaclasspython

2024-05-20 09:26:42

Python裝飾器函數

2014-03-12 10:19:54

iOS對象

2009-12-18 13:34:09

Ruby metacl

2011-08-08 09:22:10

Python

2009-08-27 13:37:11

C#類和結構

2011-07-15 13:49:30

C++友元函數友元類

2017-07-21 09:40:35

Python類、繼承和多態

2021-05-14 07:15:23

UsertpUser工具

2009-06-16 17:54:38

Scala類語法語義

2015-03-23 09:33:43

Java抽象類Java接口Java

2015-08-06 15:13:49

runtimeIOS開發

2023-12-16 13:21:00

Python元類ORM

2021-03-22 10:20:04

Python元類代碼

2013-11-11 09:34:51

超6類7類網線

2024-07-30 11:29:09

2021-04-21 10:01:53

Python類方法靜態方法

2020-10-23 07:43:31

String

2023-12-07 09:07:58

2009-07-08 17:42:26

this屬性
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 伊人伊成久久人综合网站 | 精品视频国产 | 亚洲一区二区三区免费在线观看 | 亚洲三区在线播放 | 久久青草av| av一区二区在线观看 | 一区二区精品视频 | 性色av香蕉一区二区 | 国产一区二区在线免费观看 | 99在线免费视频 | 国产精品综合视频 | 中文在线播放 | 91一区 | 久久精品国产亚洲 | 国产女人与拘做受免费视频 | 欧美国产精品一区二区三区 | 一级黄片一级毛片 | 日韩av在线中文字幕 | 精品国产一区二区三区性色av | 97人人干 | 在线观看久草 | 三级av网址| 91精品国产综合久久香蕉922 | 国产亚洲精品久久午夜玫瑰园 | 中文字幕国产日韩 | 日韩精品在线网站 | 五月激情婷婷网 | 黑人巨大精品欧美一区二区免费 | 福利视频网 | caoporn视频| 一区精品视频在线观看 | 超碰在线播 | 成人网av| 中文字幕精品一区二区三区精品 | 日韩中文字幕在线播放 | 欧美另类视频 | 伊人网在线综合 | 天天操天天操 | 欧美日韩中文字幕 | 精品亚洲一区二区三区 | 91天堂网|