国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Python Metaclass 初探

nidaye / 2326人閱讀

摘要:以我們的程序為例,就是以為產生了一個名為的新類型,改類型的實現由給出,而就包含了通過返回的這個方法。從中找到這些類并一一執行測試。

先以一個大牛的一段關于Python Metapgramming的著名的話來做開頭:

Metaclasses are deeper magic than 99% of users should ever worry about. If you wonder whether you need them, you don’t (the people who actually need them know with certainty that they need them, and don’t need an explanation about why). – Tim Peters

翻譯一下:Metaclasses是99%的用戶都無需費神的黑科技。如果你還在糾結你是不是需要它的話,答案是NO (真正需要的人根本不需要解釋) - Tim Peters

這是什么鬼話?道可道,非常道嗎?

Meta?

好,裝B已畢。這確實是一個冷僻的,不常用的話題。一篇短文肯定講不完。 所以叫做初探。

英文meta這個詞其實是從希臘語里面借來的。wikipedia上的解釋是:

indicate a concept which is an abstraction behind another concept, used to complete or add to the latter

不看還好,其實看了更暈。好在后面的解釋有一句“更高一層的抽象”,可以幫助理解。 其實我們可以這樣理解。meta的意思就是“關于什么的什么”:比如metadata可以理解為“關于數據的數據”,metaprogramming可以理解為“關于編程的編程”。這就和“更高一層的抽象” 比較契合了。同時又隱隱和編程中的另一個永恒主題-recursion聯系在了一起。

另外,meta這個詞天朝這邊翻譯成“元”,海峽對岸翻譯成“后設”。其實我都不大理解從何而來。

實例

聚焦到我們今天的主題,metaprogramming就是編寫用來生成代碼的代碼。

假設我們寫了一個NB的函數,用來計算一個任意復雜的算數表達式的值:
像1+2, 3*6+10, 什么的都可以交給它去計算。這樣的函數的算法不是我們的主題,所以我們請出python自帶的大招eval(),一行就可以搞定了:

def calc(expression):
    return eval(expression)

因為輸入的可能性是無限的,所以我們肯定要好好測試一下這個函數了。假定我們想了 上百個test case。又假定我們是用unittest這個module來做測試的。這樣的測試程序一般會長成這樣:

import unittest

class TestStringMethods(unittest.TestCase):

    def test_upper(self):
        self.assertEqual("foo".upper(), "FOO")

    def test_isupper(self):
        self.assertTrue("FOO".isupper())
        self.assertFalse("Foo".isupper())

    def test_split(self):
        s = "hello world"
        self.assertEqual(s.split(), ["hello", "world"])
        # check that s.split fails when the separator is not a string
        with self.assertRaises(TypeError):
            s.split(2)

if __name__ == "__main__":
    unittest.main()

所以我們的目的就是用metaprogramming的方式來自動產生類似上面的測試類。

先上程序后解釋:

#!/usr/bin/python3
import unittest

def calc(expression):
    return eval(expression)

def add_test(name, asserts):
    def test_method(asserts):
        def fn(self):
            left, right = asserts.split("=")
            expected = str(calc(left))
            self.assertEqual(expected, right)
        return fn

    d = {"test1": test_method(asserts)}
    cls = type(name, (unittest.TestCase,), d)
    globals()[name] = cls

if __name__ == "__main__":
    for i, t in enumerate([
            "1+2=3",
            "3*5*6=90"]):
        add_test("Test%d" % i, t)
    unittest.main()

NB的calc()函數我們解釋過了。main這段也比較簡單:我們用聲明的方式定義了一組測試,然后通過unittest來執行。

有點復雜的是add_test()。我們先來看看最內層的fn(self)這個方法。邏輯上,它就是把輸入的測試用例分成兩份,一份是calc()的輸入,一份是我們期待的結果;然后調用calc(), 接著用assertEqual()來測試。

但是這個self有點奇怪 - 這里沒有類,哪里來的self? 其實fn(self)確實是一個類的方法,只不過這個類是我們通過代碼動態生成的。也就是下面這一行:

cls = type(name, (unittest.TestCase,), d)

這里的type()就是通常我們用來檢查某個變量的類型的那個函數。只不過它還有另外一種不大為人知的形式:

class type(name, bases, dict)

這第二種形式,就會產生一個新的類型。以我們的程序為例,就是以unit.TestCase為baseclass, 產生了一個名為TestN的新類型,改類型的實現由d給出,而d就包含了通過closure返回的fn(self)這個方法。只不過在這個新類里面,它的名字叫做 test1()

最后,我們把這個新產生的類加入到當前全局符號表里面,也就相當于上面給出的unittest的例子。

所以,總結一下。當我們運行這個腳本的時候,這段比較短的代碼會針對每一個測試的表達式產生一個新的測試類,并動態生成測試的方法加載到該類里面。unitestglobals中找到這些類并一一執行測試。

上面的例子中,其實一行一行手打calc(1+2) == 3也沒什么大不了的。但是當你要表達的邏輯比較復雜的時候,metaprogramming的強大就體現出來了。

總結

那么,看完這篇文章,我們也成為Tim所說的1%的程序猿了!其實,也許他的意思是,99%的編程工作都用不到這樣技巧。在一些特殊的場合,比如編寫某種框架的時候,metaprogramming會做到事半功倍。祝你在實踐中碰到這樣的機會。

文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。

轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/40855.html

相關文章

  • [譯]理解pythonmetaclass

    摘要:但我并不是一個翻譯者并不會嚴格遵守每行每句的翻譯有時候我會將表述換個順序省略一些我認為無關緊要的話,以便讀者更好理解。類也是對象在理解之前,我們先要掌握中的類是什么。這個對象類自身擁有產生對象實例的能力。不過這樣做是為了保持向后兼容性。 前言 這篇博客是我在stackoverflow上看了一個提問回復后寫的,例子基本用的都是e-satis本人的例子,語言組織也基本按照翻譯來。 但我并不...

    liuchengxu 評論0 收藏0
  • How does it work - with_metaclass

    摘要:先簡單介紹下中的元類。元類就是創建類的類,對于元類來說,類是它的實例,將返回。中的所有類,都是的實例,換句話說,是元類的基類。 我在看源代碼的時候,經常蹦出這一句:How does it work!竟然有這種操作?本系列文章,試圖剖析代碼中發生的魔法。順便作為自己的閱讀筆記,以作提高。 先簡單介紹下Python中的元類(metaclass)。元類就是創建類的類,對于元類來說,類是它的實...

    testbird 評論0 收藏0
  • python 類和元類(metaclass)的理解和簡單運用

    摘要:什么是元類剛才說了,元類就是創建類的類。類上面的屬性,相信愿意了解元類細節的盆友,都肯定見過這個東西,而且為之好奇。使用了這個魔法方法就意味著就會用指定的元類來創建類了。深刻理解中的元類 (一) python中的類 今天看到一篇好文,然后結合自己的情況總結一波。這里討論的python類,都基于python2.7x以及繼承于object的新式類進行討論。 首先在python中,所有東西都...

    zhangqh 評論0 收藏0
  • Python: metaclass小記

    摘要:最前面那個,解釋器實際的流程是解析這段代碼,得知它需要創建一個類對象,這個類的名字叫做它的父類列表用表示是,它的屬性用一個來表示就是。解決方法很簡單關鍵就是前面被特別標記了的應當返回這個的父類的方法返回的對象。 (原發于我的blog:Python: metaclass小記 ) 友情提示:本文不一定適合閱讀,如果執意要讀,請備好暈車藥。 題記 Metaclasses are deepe...

    mushang 評論0 收藏0
  • Python -- 元類metaclass詳解

    摘要:原鏈接中的元類是什么類也是對象在理解元類之前,需要掌握中類概念。事實上,是中用于創建所有類的元類。類本身是元類的對象在中,除了,一切皆對象,一切都是類或者元類的對象。事實上是自己的元類, 學習契機 項目中使用Elasticsearch(ES)存儲海量業務數據,基于ES向外提供的API進一層封裝,按需處理原始數據提供更精確、更多樣化的結果。在研究這一層的代碼時接觸到@six.add_me...

    tracy 評論0 收藏0

發表評論

0條評論

nidaye

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<