Python基礎(chǔ)之元編程知識總結(jié)
一、前言
首先說,Python中一切皆對象,老生常談。還有,Python提供了許多特殊方法、元類等等這樣的“元編程”機(jī)制。像給對象動(dòng)態(tài)添加屬性方法之類的,在Python中根本談不上是“元編程”,但在某些靜態(tài)語言中卻是需要一定技巧的東西。我們來談些Python程序員也容易被搞糊涂的東西。
我們先來把對象分分層次,通常我們知道一個(gè)對象有它的類型,老早以前Python就將類型也實(shí)現(xiàn)為對象。這樣我們就有了實(shí)例對象和類對象。這是兩個(gè)層次。稍有基礎(chǔ)的讀者就會知道還有元類這個(gè)東西的存在,簡言之,元類就是“類”的“類”,也就是比類更高層次的東西。這又有了一個(gè)層次。還有嗎?
二、ImportTime vs RunTime
如果我們換個(gè)角度,不用非得和之前的三個(gè)層次使用同樣的標(biāo)準(zhǔn)。我們再來區(qū)分兩個(gè)東西:ImportTime和RunTime,它們之間也并非界限分明,顧名思義,就是兩個(gè)時(shí)刻,導(dǎo)入時(shí)和運(yùn)行時(shí)。
當(dāng)一個(gè)模塊被導(dǎo)入時(shí),會發(fā)生什么?在全局作用域的語句(非定義性語句)被執(zhí)行。函數(shù)定義呢?一個(gè)函數(shù)對象被創(chuàng)建,但其中的代碼不會被執(zhí)行。類定義呢?一個(gè)類對象被創(chuàng)建,類定義域的代碼被執(zhí)行,類的方法中的代碼自然也不會被執(zhí)行。
執(zhí)行時(shí)呢?函數(shù)和方法中的代碼會被執(zhí)行。當(dāng)然你要先調(diào)用它們。
三、元類
所以我們可以說元類和類是屬于ImportTime的,import一個(gè)模塊之后,它們就會被創(chuàng)建。實(shí)例對象屬于RunTime,單import是不會創(chuàng)建實(shí)例對象的。不過話不能說的太絕對,因?yàn)槿绻阋窃谀K作用域?qū)嵗悾瑢?shí)例對象也是會被創(chuàng)建的。只不過我們通常把它們寫在函數(shù)里面,所以這樣劃分。
如果你想控制產(chǎn)生的實(shí)例對象的特性該怎么做?太簡單了,在類定義中重寫__init__方法。那么我們要控制類的一些性質(zhì)呢?有這種需求嗎?當(dāng)然有!
經(jīng)典的單例模式,大家都知道有很多種實(shí)現(xiàn)方式。要求就是,一個(gè)類只能有一個(gè)實(shí)例。
最簡單的實(shí)現(xiàn)方法是這樣的
class _Spam: def __init__(self): print("Spam!!!") _spam_singleton =None def Spam(): global _spam_singleton if _spam_singleton is not None: return _spam_singleton else: _spam_singleton = _Spam() return _spam_singleton
工廠模式,不太優(yōu)雅。我們再來審視一下需求,要一個(gè)類只能有一個(gè)實(shí)例。我們在類中定義的方法都是實(shí)例對象的行為,那么要想改變類的行為,就需要更高層次的東西。元類在這個(gè)時(shí)候登場在合適不過了。前面說過,元類是類的類。也就是說,元類的__init__
方法就是類的初始化方法。 我們知道還有__call__
這個(gè)東西,它能讓實(shí)例像函數(shù)那樣被調(diào)用,那么元類的這個(gè)方法就是類在被實(shí)例化時(shí)調(diào)用的方法。
代碼就可以寫出來了:
class Singleton(type): def __init__(self, *args, **kwargs): self._instance = None super().__init__(*args, **kwargs) def __call__(self, *args, **kwargs): if self._instance is None: self._instance = super().__call__(*args, **kwargs) return self._instance else: return self._instance class Spam(metaclass=Singleton): def __init__(self): print("Spam!!!")
主要有兩個(gè)地方和一般的類定義不同,一是Singleton的基類是type,一是Spam定義的地方有一個(gè)metaclass=Singleton。type是什么?它是object的子類,object是它的實(shí)例。也就是說,type是所有類的類,也就是最基本的元類,它規(guī)定了一些所有類在產(chǎn)生時(shí)需要的一些操作。所以我們的自定義元類需要子類化type。同時(shí)type也是一個(gè)對象,所以它又是object的子類。有點(diǎn)不太好理解,大概知道就可以了。
四、裝飾器
我們再來說說裝飾器。大多數(shù)人認(rèn)為裝飾器是Python里面最難理解的概念之一。其實(shí)它不過就是一個(gè)語法糖,理解了函數(shù)也是對象之后。就可以很輕易的寫出自己的裝飾器了。
from functools import wraps def print_result(func): @wraps(func) def wrappper(*args, **kwargs): result = func(*args, **kwargs) print(result) return result return wrappper @print_result def add(x, y): return x + y #相當(dāng)于: #add = print_result(add) add(1, 3)
這里我們還用到了一個(gè)裝飾器@wraps,它是用來讓我們返回的內(nèi)部函數(shù)wrapper和原來的函數(shù)擁有相同的函數(shù)簽名的,基本上我們在寫裝飾器時(shí)都要加上它。
在注釋里寫了,@decorator這樣的形式等價(jià)于func=decorator(func),理解了這一點(diǎn),我們就可以寫出更多種類的裝飾器。比如類裝飾器,以及將裝飾器寫成一個(gè)類。
def attr_upper(cls): for attrname,value in cls.__dict__.items(): if isinstance(value,str): if not value.startswith('__'): setattr(cls,attrname,bytes.decode(str.encode(value).upper())) return cls @attr_upper class Person: sex = 'man' print(Person.sex) # MAN
注意普通的裝飾器和類裝飾器實(shí)現(xiàn)的不同點(diǎn)。
五、對數(shù)據(jù)的抽象–描述符
如果我們想讓某一些類擁有某些相同的特性,或者說可以實(shí)現(xiàn)在類定義對其的控制,我們可以自定義一個(gè)元類,然后讓它成為這些類的元類。如果我們想讓某一些函數(shù)擁有某些相同的功能,又不想把代碼復(fù)制粘貼一遍,我們可以定義一個(gè)裝飾器。那么,假如我們想讓實(shí)例的屬性擁有某些共同的特點(diǎn)呢?有人可能會說可以用property,當(dāng)然可以。但是這些邏輯必須在每個(gè)類定義的時(shí)候都寫一遍。如果我們想讓這些類的實(shí)例的某些屬性都有相同的特點(diǎn)的話,就可以自定義一個(gè)描述符類。
這里我們給出一些例子
class TypedField: def __init__(self, _type): self._type = _type def __get__(self, instance, cls): if instance is None: return self else: return getattr(instance, self.name) def __set_name__(self, cls, name): self.name = name def __set__(self, instance, value): if not isinstance(value, self._type): raise TypeError('Expected' + str(self._type)) instance.__dict__[self.name] = value class Person: age = TypedField(int) name = TypedField(str) def __init__(self, age, name): self.age = age self.name = name jack = Person(15, 'Jack') jack.age = '15' # 會報(bào)錯(cuò)
在這里面有幾個(gè)角色,TypedField
是一個(gè)描述符類,的屬性Person
是描述符類的實(shí)例,看似描述符是作為Person,也就是類的屬性而不是實(shí)例屬性存在的。但實(shí)際上,一旦Person的實(shí)例訪問了同名的屬性,描述符就會起作用。需要注意的是,在Python3.5
及之前的版本中,是沒有__set_name__
這個(gè)特殊方法的,這意味著如果你想要知道在類定義中描述符被起了一個(gè)什么樣的名字,是需要在描述符實(shí)例化時(shí)顯式傳遞給它的,也就是需要多一個(gè)參數(shù)。不過在Python3.6中,這個(gè)問題得到了解決,只需要在描述符類定義中重寫__set_name__
這個(gè)方法就好了。還需要注意的是__get__
的寫法,基本上對instance的判斷是必需的,不然會報(bào)錯(cuò)。原因也不難理解,就不細(xì)說了。
六、控制子類的創(chuàng)建——代替元類的方法
在Python3.6中,我們可以通過實(shí)現(xiàn)__init_subclass__特殊方法,來自定義子類的創(chuàng)建,這樣我們就可以在某些情況下擺脫元類這個(gè)討厭的東西。
class PluginBase: subclasses = [] def __init_subclass__(cls, **kwargs): super().__init_subclass__(**kwargs) cls.subclasses.append(cls) class Plugin1(PluginBase): pass class Plugin2(PluginBase): pass
小結(jié)諸如元類等元編程對于大多數(shù)人來說有些晦澀難懂,大多數(shù)時(shí)候也無需用到它們。但是大多數(shù)框架背后的實(shí)現(xiàn)都使用到了這些技巧,這樣才能讓使用者寫出來的代碼簡潔易懂。如果你想更深入的了解這些技巧,可以參看一些書籍例如《Fluent Python》、《Python Cookbook》(這篇文章有的內(nèi)容就是參考了它們),或者看官方文檔中的某些章節(jié)例如上文說的描述符HowTo,還有Data Model一節(jié)等等?;蛘咧苯涌碢ython的源碼,包括用Python寫的以及CPython的源碼。
記住,只有在充分理解了它們之后再去使用,也不要是個(gè)地方就想著使用這些技巧。
到此這篇關(guān)于Python基礎(chǔ)之元編程知識總結(jié)的文章就介紹到這了,更多相關(guān)Python元編程內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python 基于Appium控制多設(shè)備并行執(zhí)行
這篇文章主要介紹了python 如何基于Appium控制多設(shè)備并行執(zhí)行,幫助大家更好的理解和學(xué)習(xí)使用python,感興趣的朋友可以了解下2021-03-03深入解析神經(jīng)網(wǎng)絡(luò)從原理到實(shí)現(xiàn)
這篇文章主要介紹了深入解析神經(jīng)網(wǎng)絡(luò)從原理到實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07Yolov5訓(xùn)練意外中斷后如何接續(xù)訓(xùn)練詳解
目標(biāo)檢測是計(jì)算機(jī)視覺上的一個(gè)重要任務(wù),下面這篇文章主要給大家介紹了關(guān)于Yolov5訓(xùn)練意外中斷后如何接續(xù)訓(xùn)練的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-03-03淺談pytorch torch.backends.cudnn設(shè)置作用
今天小編就為大家分享一篇淺談pytorch torch.backends.cudnn設(shè)置作用,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-02-02詳細(xì)解讀Python中解析XML數(shù)據(jù)的方法
這篇文章主要介紹了Python中解析XML數(shù)據(jù)的方法,是Python入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下2015-10-10基于Keras的格式化輸出Loss實(shí)現(xiàn)方式
這篇文章主要介紹了基于Keras的格式化輸出Loss實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-06-06