• 八方資訊網(wǎng)歡迎您!
    八方資訊網(wǎng)>娛樂>正文

    如何將 Python 的一個(gè)類方法變?yōu)槎鄠€(gè)方法?

    2020-03-30 09:36:22 來源: 閱讀:

    作者 | 豌豆花下貓

    責(zé)編 | 郭芮

    在 Python 中,實(shí)現(xiàn)參數(shù)化測(cè)試的幾個(gè)庫(kù),是如何做到把一個(gè)方法變成多個(gè)方法,并且將每個(gè)方法與相應(yīng)的參數(shù)綁定起來的呢?

    我們?cè)偬釤捯幌拢涸谝粋€(gè)類中,如何使用裝飾器把一個(gè)類方法變成多個(gè)類方法(或者產(chǎn)生類似的效果)?

    # 帶有一個(gè)方法的測(cè)試類
    class TestClass:
    def test_func(self):
    pass

    # 使用裝飾器,生成多個(gè)類方法
    class TestClass:
    def test_func1(self):
    pass
    def test_func2(self):
    pass
    def test_func3(self):
    pass

    Python 中裝飾器的本質(zhì)就是移花接木,用一個(gè)新的方法來替代被裝飾的方法。在實(shí)現(xiàn)參數(shù)化的過程中,我們介紹過的幾個(gè)庫(kù)到底用了什么手段/秘密武器呢?

    ddt 如何實(shí)現(xiàn)參數(shù)化?

    先回顧一下上篇文章中 ddt 庫(kù)的寫法:

    import unittest
    from ddt import ddt,data,unpack
    @ddt
    class MyTest(unittest.TestCase):
    @data((3, 1), (-1, 0), (1.2, 1.0))
    @unpack
    def test(self, first, second):
    pass

    ddt 可提供 4 個(gè)裝飾器:1 個(gè)加在類上的 @ddt,還有 3 個(gè)加在類方法上的 @data、@unpack 和 @file_data(前文未提及)。

    先看看加在類方法上的三個(gè)裝飾器的作用:

    # ddt 版本(win):1.2.1
    def data(*values):
    global index_len
    index_len = len(str(len(values)))
    return idata(values)

    def idata(iterable):
    def wrapper(func):
    setattr(func, DATA_ATTR, iterable)
    return func
    return wrapper

    def unpack(func):
    setattr(func, UNPACK_ATTR, True)
    return func

    def file_data(value):
    def wrapper(func):
    setattr(func, FILE_ATTR, value)
    return func
    return wrapper

    它們的共同作用是在類方法上 setattr 添加屬性。至于這些屬性在什么時(shí)候使用?下面看看加在類上的 @ddt 裝飾器源碼:

    第一層 for 循環(huán)遍歷了所有的類方法,然后是 if/elif 兩條分支,分別對(duì)應(yīng) DATA_ATTR/FILE_ATTR,即對(duì)應(yīng)參數(shù)的兩種來源:數(shù)據(jù)(@data)和文件(@file_data)。

    elif 分支有解析文件的邏輯,之后跟處理數(shù)據(jù)相似,所以我們把它略過,主要看前面的 if 分支。這部分的邏輯很清晰,主要完成的任務(wù)如下:

    • 遍歷類方法的參數(shù)鍵值對(duì)

    • 根據(jù)原方法及參數(shù)對(duì),創(chuàng)建新的方法名

    • 獲取原方法的文檔字符串

    • 對(duì)元組和列表類型的參數(shù)作解包

    • 在測(cè)試類上添加新的測(cè)試方法,并綁定參數(shù)與文檔字符串

    分析源碼,可以看出,@data、@unpack 和 @file_data 這三個(gè)裝飾器主要是設(shè)置屬性并傳參,而 @ddt 裝飾器才是核心的處理邏輯。

    這種將裝飾器分散(分別加在類與類方法上),再組合使用的方案,很不優(yōu)雅。為什么就不能統(tǒng)一起來使用呢?后面我們會(huì)分析它的難言之隱,先按下不表,看看其它的實(shí)現(xiàn)方案是怎樣的?

    parameterized 如何實(shí)現(xiàn)參數(shù)化?

    先回顧一下上篇文章中 parameterized 庫(kù)的寫法:

    import unittest
    from parameterized import parameterized
    class MyTest(unittest.TestCase):
    @parameterized.expand([(3,1), (-1,0), (1.5,1.0)])
    def test_values(self, first, second):
    self.assertTrue(first > second)

    它提供了一個(gè)裝飾器類 @parameterized,源碼如下(版本 0.7.1),主要做了一些初始的校驗(yàn)和參數(shù)解析,并非我們關(guān)注的重點(diǎn),略過。

    我們主要關(guān)注這個(gè)裝飾器類的 expand 方法,它的文檔注釋中寫到:

    A "brute force" method of parameterizing test cases. Creates new test cases and injects them into the namespace that the wrapped function is being defined in. Useful for parameterizing tests in subclasses of 'UnitTest', where Nose test generators don't work.

    關(guān)鍵的兩個(gè)動(dòng)作是:“creates new test cases(創(chuàng)建新的測(cè)試單元)”和“inject them into the namespace…(注入到原方法的命名空間)”。

    關(guān)于第一點(diǎn),它跟 ddt 是相似的,只是一些命名風(fēng)格上的差異,以及參數(shù)的解析及綁定不同,不值得太關(guān)注

    最不同的則是,怎么令新的測(cè)試方法生效?

    parameterized 使用的是一種“注入”的方式:

    inspect 是個(gè)功能強(qiáng)大的標(biāo)準(zhǔn)庫(kù),在此用于獲取程序調(diào)用棧的信息。前三句代碼的目的是取出 f_locals,它的含義是“l(fā)ocal namespace seen by this frame”,此處 f_locals 指的就是類的局部命名空間。

    說到局部命名空間,你可能會(huì)想到 locals,但是,我們之前有文章提到過“l(fā)ocals 與 globals 的讀寫問題”,locals 是可讀不可寫的,所以這段代碼才用了 f_locals。

    pytest 如何實(shí)現(xiàn)參數(shù)化?

    按慣例先看看上篇文章中的寫法:

    import pytest
    @pytest.mark.parametrize("first,second", [(3,1), (-1,0), (1.5,1.0)])
    def test_values(first, second):
    assert(first > second)

    首先看到“mark”,pytest 里內(nèi)置了一些標(biāo)簽,例如 parametrize、timeout、skipif、xfail、tryfirst、trylast 等,還支持用戶自定義的標(biāo)簽,可以設(shè)置執(zhí)行條件、分組篩選執(zhí)行,以及修改原測(cè)試行為等等。

    用法也是非常簡(jiǎn)單的,然而,其源碼可復(fù)雜多了。我們這里只關(guān)注 parametrize,先看看核心的一段代碼:

    根據(jù)傳入的參數(shù)對(duì),它復(fù)制了原測(cè)試方法的調(diào)用信息,存入待調(diào)用的列表里。跟前面分析的兩個(gè)庫(kù)不同,它并沒有在此創(chuàng)建新的測(cè)試方法,而是復(fù)用了已有的方法。在 parametrize 所屬的 Metafunc 類往上查找,可以追蹤到 _calls 列表的使用位置:

    最終是在 Function 類中執(zhí)行:

    好玩的是,在這里我們可以看到幾行神注釋……

    閱讀(粗淺涉獵) pytest 的源碼,真的是自討苦吃……不過,依稀大致可以看出,它在實(shí)現(xiàn)參數(shù)化時(shí),使用的是生成器的方案,遍歷一個(gè)參數(shù)則調(diào)用一次測(cè)試方法,而前面的 ddt 和 parameterized 則是一次性把所有參數(shù)解析完,生成 n 個(gè)新的測(cè)試方法,再交給測(cè)試框架去調(diào)度。

    對(duì)比一下,前兩個(gè)庫(kù)的思路很清晰,而且由于其設(shè)計(jì)單純是為了實(shí)現(xiàn)參數(shù)化,不像 pytest 有什么標(biāo)記和過多的抽象設(shè)計(jì),所以更易讀易懂。前兩個(gè)庫(kù)發(fā)揮了 Python 的動(dòng)態(tài)特性,設(shè)置類屬性或者注入局部命名空間,而 pytest 倒像是從什么靜態(tài)語(yǔ)言中借鑒的思路,略顯笨拙。

    小結(jié)

    回到標(biāo)題中的問題“如何將一個(gè)方法變?yōu)槎鄠€(gè)方法?”除了在參數(shù)化測(cè)試中,不知還有哪些場(chǎng)景會(huì)有此訴求?歡迎留言討論。

    本文分析了三個(gè)測(cè)試庫(kù)的裝飾器實(shí)現(xiàn)思路,通過閱讀源碼,我們可以發(fā)現(xiàn)它們各有千秋,這個(gè)發(fā)現(xiàn)本身還挺有意思。在使用裝飾器時(shí),表面看它們差異不大,但是真功夫的細(xì)節(jié)都隱藏在底下。

    作者:豌豆花下貓,生于廣東畢業(yè)于武大,現(xiàn)為蘇漂程序員,有一些極客思維,也有一些人文情懷,有一些溫度,還有一些態(tài)度,公眾號(hào)「Python貓」(python_cat)。

    聲明:本文系作者投稿,版權(quán)歸作者所有。

    本文為企業(yè)推廣,本網(wǎng)站不做任何建議,僅提供參考,作為信息展示!

    推薦閱讀:iphone7對(duì)比iphone8

    網(wǎng)友評(píng)論
    請(qǐng)登錄后進(jìn)行評(píng)論| 0條評(píng)論

    請(qǐng)文明發(fā)言,還可以輸入140

    您的評(píng)論已經(jīng)發(fā)表成功,請(qǐng)等候?qū)徍?/p>

    小提示:您要為您發(fā)表的言論后果負(fù)責(zé),請(qǐng)各位遵守法紀(jì)注意語(yǔ)言文明

    回到首頁(yè) 回到頂部
    八方資訊網(wǎng) 關(guān)于我們| 聯(lián)系我們| 招聘信息| 老版地圖| 網(wǎng)站地圖
    免責(zé)聲明:八方資訊網(wǎng)所有文字、圖片、視頻、音頻等資料均來自互聯(lián)網(wǎng),不代表本站贊同其觀點(diǎn),本站亦不為其版權(quán)負(fù)責(zé)。相關(guān)作品的原創(chuàng)性、文中陳述文字以及內(nèi)容數(shù)據(jù)龐雜本站無法一一核實(shí),如果您發(fā)現(xiàn)本網(wǎng)站上有侵犯您的合法權(quán)益的內(nèi)容,請(qǐng)聯(lián)系我們,本網(wǎng)站將立即予以刪除!
    Copyright © 2012-2019 http://m.quan28.cn, All rights reserved.
    主站蜘蛛池模板: 精品无人区无码乱码大片国产| 国产精品igao视频| 国亚洲欧美日韩精品| 人妻少妇看A偷人无码精品视频| 欧美精品免费专区在线观看| 精品久久久久久久中文字幕| 亚洲国产精品成人一区| 无码精品A∨在线观看| AV无码精品一区二区三区| 国产精品福利在线观看| 国产精品免费在线播放| 夜夜高潮夜夜爽国产伦精品| 久久久久久国产精品免费无码| 国产精品手机在线| 99免费精品国产| 亚洲精品国产成人影院| 国产精品无码av在线播放| 欧美精品国产日韩综合在线| 精品国内自产拍在线观看| 亚洲av永久无码精品网站| 欧美精品黑人巨大在线播放| 久久久久久亚洲精品不卡| 国产精品精品自在线拍| www.精品| 国产精品视频一区二区三区四| 午夜精品免费在线观看| 人妻精品久久久久中文字幕| 无码日韩精品一区二区三区免费| 97久久精品人妻人人搡人人玩| 国产精品嫩草影院久久| 日韩精品真人荷官无码| 国产成人精品福利网站在线观看| 热re99久久6国产精品免费| 亚洲国产精品18久久久久久| 国产成人精品免费视频大| 亚欧洲精品在线视频免费观看| 久久精品一区二区国产| 国产亚洲精品美女久久久| 亚洲色精品aⅴ一区区三区| 精品人妻少妇一区二区| 久久精品这里热有精品|