Python TDD(Testing Driven Development)詳解
coding的形成過(guò)程應(yīng)該是從special case入手,找到general solving methods麸澜,為了提高編碼效率条辟,TDD是一個(gè)不錯(cuò)的辦法晤锹。
實(shí)例:
輸入——一個(gè)自然數(shù)list
功能——按照l(shuí)ist的順序先輸出list中所有偶數(shù)凤薛,再輸出剩下的奇數(shù)
輸出——功能化后的list
1.實(shí)例實(shí)現(xiàn)
應(yīng)用之前提到的python list生成的優(yōu)化寫法:
def f(L):
return [e for e in L if e % 2 == 0] + [e for e in L if e % 2 == 1]
這種寫法效率不高收厨,因?yàn)橐?次遍歷數(shù)組L分別找偶數(shù)和奇數(shù)缴啡,可以優(yōu)化為:
def f(L):
even_member = []
odd_member = []
for e in L:
if e % 2 == 0:
even_member.append(e)
else:
odd_member.append(e)
return even_member + odd_member
2.Test設(shè)計(jì)
這個(gè)實(shí)例是關(guān)于奇偶數(shù)的排序,所以應(yīng)該包含如下的測(cè)試
(1)[]
(2)[奇數(shù)]
(3)[偶數(shù)]
(4)[奇數(shù)骂际, 奇數(shù)]
(5)[偶數(shù)疗琉,偶數(shù)]
(6)[奇數(shù),偶數(shù)]
(7)[偶數(shù)歉铝,奇數(shù)]
(8)[偶數(shù)盈简,偶數(shù),偶數(shù)太示,偶數(shù)]
(9)[奇數(shù)柠贤, 奇數(shù),奇數(shù)类缤, 奇數(shù)]
(10)[奇數(shù)臼勉,偶數(shù),偶數(shù)...]
3.原程序中人肉輸出檢驗(yàn)
使用print()函數(shù)顯示出來(lái)餐弱,人肉核對(duì)是否正確宴霸。
print(f([]))
print(f([0]))
print(f([1]))
print(f([0, 2]))
print(f([1, 3]))
print(f([0, 3]))
print(f([3, 0]))
print(f([2, 0, 8, 6]))
print(f([5, 1, 3, 9]))
print(f([2, 7, 8, 3, 4]))
4.原程序中設(shè)置檢驗(yàn)函數(shù)
人肉檢驗(yàn)的問(wèn)題不僅是傷身,更是會(huì)影響函數(shù)調(diào)用膏蚓。如果其他程序調(diào)用了一個(gè)包含這么多人肉輸出代碼的程序瓢谢,跑出來(lái)的結(jié)果會(huì)非常尷尬。設(shè)想如果你打開微信驮瞧,結(jié)果微信由于調(diào)用了這樣一個(gè)程序氓扛,導(dǎo)致你的界面上有若干測(cè)試結(jié)果,只能用災(zāi)難形容论笔。
一個(gè)小進(jìn)步是先解決程序調(diào)用問(wèn)題采郎。
def f(L):
return [e for e in L if e % 2 == 0] + [e for e in L if e % 2 == 1]
print("Name of module: ", __name__)
if __name__ == '__main__':
print('\nTesting [], result should be [].')
print(f([]) == [])
print('\nTesting [0], result should be [0].')
print(f([0]) == [0])
print('\nTesting [1], result should be [1].')
print(f([1]) == [1])
print('\nTesting [0, 1], result should be [0, 1].')
print(f([0, 1]) == [0, 1])
print('\nTesting [1, 0], result should be [0, 1].')
print(f([1, 0]) == [0, 1])
print('\nTesting [3, 0, 11, 17, 10, 11, 17, 15, 18, 5], result should be [0, 10, 18, 3, 11, 17, 11, 17, 15, 5].')
print(f([3, 0, 11, 17, 10, 11, 17, 15, 18, 5]) == [0, 10, 18, 3, 11, 17, 11, 17, 15, 5])
由于在人肉輸出的時(shí)候有了限制條件if __name__ == '__main__',含義是這個(gè)程序名稱是不是main狂魔,所以解決了剛才人肉輸出的調(diào)用問(wèn)題尉剩。
在函數(shù)調(diào)用時(shí),當(dāng)前的程序名稱是main毅臊,被調(diào)用的程序名稱是其“文件名”。這樣在調(diào)用上述函數(shù)的時(shí)候,if條件不滿足管嬉,各種檢驗(yàn)不會(huì)運(yùn)行皂林。
另外,對(duì)人肉工作做了一點(diǎn)優(yōu)化蚯撩,不再需要人肉核對(duì)所有l(wèi)ist元素础倍,而是直接給出True or False的結(jié)果。
5.寫測(cè)試程序
實(shí)際工作中胎挎,不同module是嚴(yán)格分開的沟启,測(cè)試程序一般要在新的python文件下編碼,需要測(cè)試的時(shí)候可以方便調(diào)用犹菇。
def f(L):
return [e for e in L if e % 2 == 0] + [e for e in L if e % 2 == 1]
if __name__ == '__main__':
import test
test.test()
上述程序在源程序相同路徑下存放有一個(gè)文件名為"test"的測(cè)試程序德迹,import后運(yùn)行其中的test()函數(shù)。這里注意揭芍,不要忘記寫if條件胳搞,否則調(diào)用的時(shí)候可能會(huì)由于路徑問(wèn)題報(bào)錯(cuò)。
6.Python prompt測(cè)試
分開寫測(cè)試文件確實(shí)常見(jiàn)称杨,但是對(duì)于小的函數(shù)未免太過(guò)麻煩肌毅,所以最推薦的做法是利用python prompt做測(cè)試。
def f(L):
'''
>>> f([])
[]
>>> f([0, 1])
[0, 1]
>>> f([1, 0])
[1, 0]
'''
return [e for e in L if e % 2 == 0] + [e for e in L if e % 2 == 1]
import doctest
doctest.testmod()
寫法是在def函數(shù)下一行縮減后姑原,寫入多行注釋悬而。在注釋中,>>>代表python的prompt锭汛,后面寫的是設(shè)計(jì)好的測(cè)試笨奠,換行后給出預(yù)期的結(jié)果。如果結(jié)果不同店乐,python運(yùn)行時(shí)會(huì)報(bào)錯(cuò)艰躺。