前言
前面講到unittest裏面setUp能夠在每次執行用例前執行,這樣有效的減小了代碼量,可是有個弊端,好比打開瀏覽器操做,每次執行用例時候都會從新打開,這樣就會浪費不少時間。
因而就想是否是能夠只打開一次瀏覽器,執行完用例再關閉呢?這就須要用到裝飾器(@classmethod)來解決了。
1、裝飾器
1.用setUp與setUpClass區別python
setup():每一個測試case運行前運行
teardown():每一個測試case運行完後執行
setUpClass():必須使用@classmethod 裝飾器,全部case運行前只運行一次
tearDownClass():必須使用@classmethod裝飾器,全部case運行完後只運行一次web
2.@是修飾符,classmethod是python裏的類方法瀏覽器
2、執行順序
1.用類方法寫幾個簡單case,能夠對比這篇:Selenium2+python自動化52-unittest執行順序框架
# coding:utf-8 import unittest import time
class Test(unittest.TestCase): @classmethod def setUpClass(cls): print "start!" @classmethod def tearDownClass(cls): time.sleep(1) print "end!" def test01(self): print "執行測試用例01" def test03(self): print "執行測試用例03" def test02(self): print "執行測試用例02" def addtest(self): print "add方法"
if __name__ == "__main__": unittest.main()
2.從執行結果能夠看出,前置和後置在執行用例前只執行了一次。
start!
執行測試用例01
執行測試用例02
執行測試用例03
...end!less
----------------------------------------------------------------------
Ran 3 tests in 1.001s函數
3、selenium實例
1.能夠把打開瀏覽器操做放到前置setUpClass(cls)裏,這樣就能夠實現打開一次瀏覽器,執行多個case了測試
# coding:utf-8 from selenium import webdriver from selenium.webdriver.support import expected_conditions as EC import unittest
class BolgHome(unittest.TestCase):
u'''博客首頁''' @classmethod def setUpClass(cls): cls.driver = webdriver.Firefox() url = "http://www.cnblogs.com/yoyoketang/" cls.driver.get(url) cls.driver.implicitly_wait(30)
@classmethod def tearDownClass(cls): cls.driver.quit()
def test_01(self): u'''驗證元素存在:博客園''' locator = ("id", "blog_nav_sitehome") text = u"博客園" result = EC.text_to_be_present_in_element(locator, text)(self.driver) self.assertTrue(result) def test_02(self): u'''驗證元素存在:首頁''' locator = ("id", "blog_nav_myhome") text = u"首頁" result = EC.text_to_be_present_in_element(locator, text)(self.driver) self.assertTrue(result)
if __name__ == "__main__": unittest.main()
上述代碼運行的順序就是從上至下,而再也不是每次執行完成一個testcase以後,執行一次teardownClass再進行下一個testcase。ui
這樣一來,退出瀏覽器僅僅執行一次便可,這樣有一個很差的地方就是,teardownClass這個函數不能再進行每一個測試用例的終結操做,好比:修改我的信息後恢復到登陸成功後的狀態,對當前測試用例的異常處理等。url
後來,本人嘗試在tearDownClass後增長以下代碼:spa
def tearDown(self): self.driver.refresh() self.assertEqual( [], self.verificationErrors )
而後,果真每次測試用完成都會刷新當前頁面,這樣一來,每個testcase的用例都能被終結函數tearDown結束,最後再執行tearDownClass關閉測試瀏覽器。
須要說明的是:
@classmethod是python自己的裝飾器,因此他不要使用隸屬於unittest框架下斷言assertEqual。
unittest自己也帶有裝飾器unittest.skip(),專門用於跳過testcase的裝飾器,其用法以下:
@unittest.skip(reason), skip裝飾器:直接跳過裝飾下的testcase,reason用來講明緣由,下同。 @unittest.skipIf(condition,reason), skipIf裝飾器:condition條件爲True時,跳過裝飾下的testcase,計入skip的testcase執行次數。 @unittest.skipUnless(condition,reason),skipUnless裝飾器:condition條件爲False時,跳過裝飾下的testcase,計入skip的testcase執行次數。 @unittest.expectedFailure(), expectedFailure裝飾器:執行裝飾下的testcase,執行失敗則跳過該testcase,計入expected下成敗的testcase次數。
通常來說,使用@unittest.skipIf 或者 @unittest.skipUnless,應該也能實現@classmethod裝飾器的效果, 想來只是實現起來相對來講較爲麻煩。