0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Python里的元編程:控制產(chǎn)生的實(shí)例對(duì)象的特性以及實(shí)例

馬哥Linux運(yùn)維 ? 2017-12-26 18:01 ? 次閱讀

很多人不理解“元編程”是個(gè)什么東西,關(guān)于它也沒(méi)有一個(gè)十分準(zhǔn)確的定義。這篇文章要說(shuō)的是Python里的元編程,實(shí)際上也不一定就真的符合“元編程”的定義。只不過(guò)我無(wú)法找到一個(gè)更準(zhǔn)確的名字來(lái)代表這篇文章的主題,所以就借了這么一個(gè)名號(hào)。

副標(biāo)題是控制你想控制的一切,實(shí)際上這篇文章講的都是一個(gè)東西,利用Python提供給我們的特性,盡可能的使代碼優(yōu)雅簡(jiǎn)潔。具體而言,通過(guò)編程的方法,在更高的抽象層次上對(duì)一種層次的抽象的特性進(jìn)行修改。

首先說(shuō),Python中一切皆對(duì)象,老生常談。還有,Python提供了許多特殊方法、元類(lèi)等等這樣的“元編程”機(jī)制。像給對(duì)象動(dòng)態(tài)添加屬性方法之類(lèi)的,在Python中根本談不上是“元編程”,但在某些靜態(tài)語(yǔ)言中卻是需要一定技巧的東西。我們來(lái)談些Python程序員也容易被搞糊涂的東西。

我們先來(lái)把對(duì)象分分層次,通常我們知道一個(gè)對(duì)象有它的類(lèi)型,老早以前Python就將類(lèi)型也實(shí)現(xiàn)為對(duì)象。這樣我們就有了實(shí)例對(duì)象和類(lèi)對(duì)象。這是兩個(gè)層次。稍有基礎(chǔ)的讀者就會(huì)知道還有元類(lèi)這個(gè)東西的存在,簡(jiǎn)言之,元類(lèi)就是“類(lèi)”的“類(lèi)”,也就是比類(lèi)更高層次的東西。這又有了一個(gè)層次。還有嗎?ImportTime vs RunTime

如果我們換個(gè)角度,不用非得和之前的三個(gè)層次使用同樣的標(biāo)準(zhǔn)。我們?cè)賮?lái)區(qū)分兩個(gè)東西:ImportTime和RunTime,它們之間也并非界限分明,顧名思義,就是兩個(gè)時(shí)刻,導(dǎo)入時(shí)和運(yùn)行時(shí)。

當(dāng)一個(gè)模塊被導(dǎo)入時(shí),會(huì)發(fā)生什么?在全局作用域的語(yǔ)句(非定義性語(yǔ)句)被執(zhí)行。函數(shù)定義呢?一個(gè)函數(shù)對(duì)象被創(chuàng)建,但其中的代碼不會(huì)被執(zhí)行。類(lèi)定義呢?一個(gè)類(lèi)對(duì)象被創(chuàng)建,類(lèi)定義域的代碼被執(zhí)行,類(lèi)的方法中的代碼自然也不會(huì)被執(zhí)行。

執(zhí)行時(shí)呢?函數(shù)和方法中的代碼會(huì)被執(zhí)行。當(dāng)然你要先調(diào)用它們。元類(lèi)

所以我們可以說(shuō)元類(lèi)和類(lèi)是屬于ImportTime的,import一個(gè)模塊之后,它們就會(huì)被創(chuàng)建。實(shí)例對(duì)象屬于RunTime,單import是不會(huì)創(chuàng)建實(shí)例對(duì)象的。不過(guò)話不能說(shuō)的太絕對(duì),因?yàn)槿绻阋窃谀K作用域?qū)嵗?lèi),實(shí)例對(duì)象也是會(huì)被創(chuàng)建的。只不過(guò)我們通常把它們寫(xiě)在函數(shù)里面,所以這樣劃分。

如果你想控制產(chǎn)生的實(shí)例對(duì)象的特性該怎么做?太簡(jiǎn)單了,在類(lèi)定義中重寫(xiě)init方法。那么我們要控制類(lèi)的一些性質(zhì)呢?有這種需求嗎?當(dāng)然有!

經(jīng)典的單例模式,大家都知道有很多種實(shí)現(xiàn)方式。要求就是,一個(gè)類(lèi)只能有一個(gè)實(shí)例。

最簡(jiǎn)單的實(shí)現(xiàn)方法是這樣的

工廠模式,不太優(yōu)雅。我們?cè)賮?lái)審視一下需求,要一個(gè)類(lèi)只能有一個(gè)實(shí)例。我們?cè)陬?lèi)中定義的方法都是實(shí)例對(duì)象的行為,那么要想改變類(lèi)的行為,就需要更高層次的東西。元類(lèi)在這個(gè)時(shí)候登場(chǎng)在合適不過(guò)了。前面說(shuō)過(guò),元類(lèi)是類(lèi)的類(lèi)。也就是說(shuō),元類(lèi)的init方法就是類(lèi)的初始化方法。 我們知道還有call這個(gè)東西,它能讓實(shí)例像函數(shù)那樣被調(diào)用,那么元類(lèi)的這個(gè)方法就是類(lèi)在被實(shí)例化時(shí)調(diào)用的方法。

代碼就可以寫(xiě)出來(lái)了:

Python里的元編程:控制產(chǎn)生的實(shí)例對(duì)象的特性以及實(shí)例

主要有兩個(gè)地方和一般的類(lèi)定義不同,一是Singleton的基類(lèi)是type,一是Spam定義的地方有一個(gè)metaclass=Singleton。type是什么?它是object的子類(lèi),object是它的實(shí)例。也就是說(shuō),type是所有類(lèi)的類(lèi),也就是最基本的元類(lèi),它規(guī)定了一些所有類(lèi)在產(chǎn)生時(shí)需要的一些操作。所以我們的自定義元類(lèi)需要子類(lèi)化type。同時(shí)type也是一個(gè)對(duì)象,所以它又是object的子類(lèi)。有點(diǎn)不太好理解,大概知道就可以了。裝飾器

我們?cè)賮?lái)說(shuō)說(shuō)裝飾器。大多數(shù)人認(rèn)為裝飾器是Python里面最難理解的概念之一。其實(shí)它不過(guò)就是一個(gè)語(yǔ)法糖,理解了函數(shù)也是對(duì)象之后。就可以很輕易的寫(xiě)出自己的裝飾器了。

這里我們還用到了一個(gè)裝飾器@wraps,它是用來(lái)讓我們返回的內(nèi)部函數(shù)wrapper和原來(lái)的函數(shù)擁有相同的函數(shù)簽名的,基本上我們?cè)趯?xiě)裝飾器時(shí)都要加上它。

我在注釋里寫(xiě)了,@decorator這樣的形式等價(jià)于func=decorator(func),理解了這一點(diǎn),我們就可以寫(xiě)出更多種類(lèi)的裝飾器。比如類(lèi)裝飾器,以及將裝飾器寫(xiě)成一個(gè)類(lèi)。

Python里的元編程:控制產(chǎn)生的實(shí)例對(duì)象的特性以及實(shí)例

注意普通的裝飾器和類(lèi)裝飾器實(shí)現(xiàn)的不同點(diǎn)。對(duì)數(shù)據(jù)的抽象--描述符

如果我們想讓某一些類(lèi)擁有某些相同的特性,或者說(shuō)可以實(shí)現(xiàn)在類(lèi)定義對(duì)其的控制,我們可以自定義一個(gè)元類(lèi),然后讓它成為這些類(lèi)的元類(lèi)。如果我們想讓某一些函數(shù)擁有某些相同的功能,又不想把代碼復(fù)制粘貼一遍,我們可以定義一個(gè)裝飾器。那么,假如我們想讓實(shí)例的屬性擁有某些共同的特點(diǎn)呢?有人可能會(huì)說(shuō)可以用property,當(dāng)然可以。但是這些邏輯必須在每個(gè)類(lèi)定義的時(shí)候都寫(xiě)一遍。如果我們想讓這些類(lèi)的實(shí)例的某些屬性都有相同的特點(diǎn)的話,就可以自定義一個(gè)描述符類(lèi)。

這里我們給出一些例子。

Python里的元編程:控制產(chǎn)生的實(shí)例對(duì)象的特性以及實(shí)例

在這里面有幾個(gè)角色,TypedField是一個(gè)描述符類(lèi),Person的屬性是描述符類(lèi)的實(shí)例,看似描述符是作為Person,也就是類(lèi)的屬性而不是實(shí)例屬性存在的。但實(shí)際上,一旦Person的實(shí)例訪問(wèn)了同名的屬性,描述符就會(huì)起作用。需要注意的是,在Python3.5及之前的版本中,是沒(méi)有set_name這個(gè)特殊方法的,這意味著如果你想要知道在類(lèi)定義中描述符被起了一個(gè)什么樣的名字,是需要在描述符實(shí)例化時(shí)顯式傳遞給它的,也就是需要多一個(gè)參數(shù)。不過(guò)在Python3.6中,這個(gè)問(wèn)題得到了解決,只需要在描述符類(lèi)定義中重寫(xiě)set_name這個(gè)方法就好了。還需要注意的是get的寫(xiě)法,基本上對(duì)instance的判斷是必需的,不然會(huì)報(bào)錯(cuò)。原因也不難理解,就不細(xì)說(shuō)了??刂谱宇?lèi)的創(chuàng)建——代替元類(lèi)的方法

在Python3.6中,我們可以通過(guò)實(shí)現(xiàn)init_subclass特殊方法,來(lái)自定義子類(lèi)的創(chuàng)建,這樣我們就可以在某些情況下擺脫元類(lèi)這個(gè)討厭的東西。

小結(jié)

諸如元類(lèi)等元編程對(duì)于大多數(shù)人來(lái)說(shuō)有些晦澀難懂,大多數(shù)時(shí)候也無(wú)需用到它們。但是大多數(shù)框架背后的實(shí)現(xiàn)都使用到了這些技巧,這樣才能讓使用者寫(xiě)出來(lái)的代碼簡(jiǎn)潔易懂。如果你想更深入的了解這些技巧,可以參看一些書(shū)籍例如《Fluent Python》、《Python Cookbook》(這篇文章有的內(nèi)容就是參考了它們),或者看官方文檔中的某些章節(jié)例如上文說(shuō)的描述符HowTo,還有Data Model一節(jié)等等。或者直接看Python的源碼,包括用Python寫(xiě)的以及CPython的源碼。

記住,只有在充分理解了它們之后再去使用,也不要是個(gè)地方就想著使用這些技巧。

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • python
    +關(guān)注

    關(guān)注

    56

    文章

    4807

    瀏覽量

    85012

原文標(biāo)題:Python元編程:控制你想控制的一切

文章出處:【微信號(hào):magedu-Linux,微信公眾號(hào):馬哥Linux運(yùn)維】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    Python的面向對(duì)象編程詳解

    一般編程可分為面向過(guò)程編程,和面向對(duì)象編程。Python的面向對(duì)象
    發(fā)表于 09-04 16:35 ?591次閱讀
    <b class='flag-5'>Python</b>的面向<b class='flag-5'>對(duì)象</b><b class='flag-5'>編程</b>詳解

    FX編程實(shí)例

    三菱PLC(可編程邏輯控制器)編程實(shí)例項(xiàng)目例程
    發(fā)表于 11-07 15:24 ?1次下載

    控制溫度PID實(shí)例

    三菱PLC(可編程邏輯控制器)編程實(shí)例項(xiàng)目例程控制溫度PID實(shí)例
    發(fā)表于 05-13 16:13 ?52次下載

    控制行車(chē)_PLC編程實(shí)例項(xiàng)目例程

    三菱PLC(可編程邏輯控制器)編程實(shí)例項(xiàng)目例程
    發(fā)表于 11-05 15:01 ?6次下載

    馬達(dá)控制_PLC編程實(shí)例項(xiàng)目例程

    三菱PLC(可編程邏輯控制器)編程實(shí)例項(xiàng)目例程
    發(fā)表于 11-05 14:37 ?12次下載

    氣壓控制_PLC編程實(shí)例項(xiàng)目例程

    三菱PLC(可編程邏輯控制器)編程實(shí)例項(xiàng)目例程
    發(fā)表于 11-05 14:15 ?10次下載

    三菱編程實(shí)例

    三菱PLC(可編程邏輯控制器)編程實(shí)例項(xiàng)目例程
    發(fā)表于 12-10 14:58 ?18次下載

    Python編程實(shí)例

    Python編程實(shí)例
    發(fā)表于 01-08 14:14 ?0次下載

    python串口編程實(shí)例

    本文主要介紹了幾種python串口編程實(shí)例。Python是純粹的自由軟件, 源代碼和解釋器CPython遵循 GPL(GNU General Public License)協(xié)議。
    發(fā)表于 01-15 09:35 ?4.6w次閱讀

    面向對(duì)象與圖形編程的詳細(xì)資料和實(shí)例說(shuō)明

    本文檔的主要內(nèi)容詳細(xì)介紹的是面向對(duì)象與圖形編程的詳細(xì)資料和實(shí)例說(shuō)明資料免費(fèi)下載
    發(fā)表于 11-01 16:49 ?11次下載

    python的經(jīng)典實(shí)例相關(guān)講解

    本文檔的主要內(nèi)容詳細(xì)介紹的是python的經(jīng)典實(shí)例相關(guān)講解。
    發(fā)表于 03-02 15:33 ?9次下載

    python經(jīng)典實(shí)例詳解

    python經(jīng)典實(shí)例詳解說(shuō)明。
    發(fā)表于 04-26 10:14 ?32次下載

    利用Python編寫(xiě)簡(jiǎn)單網(wǎng)絡(luò)爬蟲(chóng)實(shí)例

    利用 Python編寫(xiě)簡(jiǎn)單網(wǎng)絡(luò)爬蟲(chóng)實(shí)例2 實(shí)驗(yàn)環(huán)境python版本:3.3.5(2.7下報(bào)錯(cuò)
    發(fā)表于 02-24 11:05 ?14次下載

    Python中的類(lèi)和對(duì)象詳解

    Python 是一種面向對(duì)象編程語(yǔ)言,它支持類(lèi)和對(duì)象。類(lèi)是一種用戶(hù)自定義的數(shù)據(jù)類(lèi)型,用于定義對(duì)象的屬性和方法。
    的頭像 發(fā)表于 04-20 16:53 ?1214次閱讀

    Python面向對(duì)象深化(下)

    __call__ ?方法和可調(diào)用對(duì)象 凡是可以將 () 直接應(yīng)用到自身并執(zhí)行, 都稱(chēng)為可調(diào)用對(duì)象. 注意: 可調(diào)用對(duì)象包括自定義的函數(shù)、Python 內(nèi)置函數(shù)、
    的頭像 發(fā)表于 05-11 17:52 ?690次閱讀