面向對象的特性:封裝、繼承、多態。在自動化中一樣適用,Selenium自動化測試中有一個名字常常被提及PageObject(思想與面向對象的特性相同),通過PO模式可以大大提高測試用例的維護效率。
##傳統測試腳本的弊端
- 測試腳本分離,維護成本高
- 可擴展性差
- 複用性低等
PageObject設計模式
PO的核心要素:
- 在PO模式中抽象封裝成一個BasePage類,該基類應該擁有一個只實現webdriver實例的屬性。
- 每個一個page都繼承BasePage,通過driver來管理本page中元素,將page中的操作封裝成一個個的方法。
- TestCase繼承unittest.Testcase類,並且依賴page類,從而實現相應的測試步驟。
案例
基礎案例
前面基礎場景選取的是baidu搜索頁面(baidu頁面簡單,不需要搭建測試環境)baidu.py
from selenium import webdriver
from time import sleep
driver = webdriver.Firefox()
driver.get("http://www.baidu.com")
driver.find_element_by_xpath("//input[@id='kw']").send_keys("Bela")
driver.find_element_by_xpath("//input[@id='su']").click()
sleep(5)
driver.quit()
將上面的腳本放在baidu.py文件中。
分析
通過對baidu.py腳本的分析,可以提取到:
- 不同的運行腳本環境,瀏覽器不同:驅動webdriver.Firefox()可以剝離,
- 請求地址的變化(生產環境與測試環境):url==http://www.baidu.com可以剝離
- 操作元素時,常常需要等待元素加載完畢後方可進行操作:是否可以把webdriver提供的findelement* 方法封裝下,才操作元素前,先判斷元素的是否可操作。
===================================================
-
實際測試場景中,可能有多個測試場景,如果每個測試場景都需要維護url、瀏覽器驅動、元素定位等,效率會非常低。
- 因此基於對上面的分析,是否可以設計一個所有測試頁面(selenium本身是對B/S系統開展測試)的基類,來維護一些公共的方法。此處先定義個名字哦BasePage.py,用於存放頁面公共方法及webdriver原有方法二次封裝等。
BasePage.py內容如下:
from selenium.webdriver.support.wait import WebDriverWait
from selenium import webdriver
from selenium.webdriver.support import expected_conditions as EC
class BasePage(object):
"""
BasePage封裝所有頁面都公用的方法,例如driver, Find_Element等
"""
# 實例化BasePage類時,最先執行的就是__init__方法,該方法的入參,其實就是BasePage類的入參。
# __init__方法不能有返回值,只能返回None
def __init__(self,selenium_driver,base_url):
self.driver = selenium_driver
self.base_url = base_url
# self.pagetitle = pagetitle
def on_page(self,pagetitle):
return pagetitle in self.driver.title
def _open(self,url):
self.driver.get(url)
self.driver.maximize_window()
def open(self):
self._open(self.base_url,self.pagetitle)
def find_element(self,*loc): #*loc任意數量的位置參數(帶單個星號參數)
# return self.driver.find_element(*loc)
try:
WebDriverWait(self.driver,10).until(EC.visibility_of_element_located(loc))
return self.driver.find_element(*loc)
except:
print("%s 頁面未能找到 %s 元素"%(self,loc))
def script(self,src):
self.driver.excute_script(src)
def send_keys(self, loc, vaule, clear_first=True, click_first=True):
try:
loc = getattr(self, "_%s" % loc) # getattr相當於實現self.loc
if click_first:
self.find_element(*loc).click()
if clear_first:
self.find_element(*loc).clear()
self.find_element(*loc).send_keys(vaule)
except AttributeError:
print("%s 頁面中未能找到 %s 元素" % (self, loc))
測試腳本的優化
BasePage.py提取完畢,其中設計了BasePage類,對一些webdriver的方法進行了二次封裝。
baidu.py基於BasePage.py進行優化(充分體現PO的設計思想,封裝、繼承)
# 基本測試場景
# from selenium import webdriver
# from time import sleep
#
# driver = webdriver.Firefox()
# driver.get("http://www.baidu.com")
#
# driver.find_element_by_xpath("//input[@id='kw']").send_keys("Bela") #輸入框
# driver.find_element_by_xpath("//input[@id='su']").click() #百度一下按鈕
#
# sleep(3)
# driver.quit()
# 優化後的測試場景
from selenium.webdriver.common.by import By
from PODemo.BasePage import BasePage #假設baidu.py、BasePage.py均在PODemo.BasePage目錄下
from selenium import webdriver
class SearchPage(BasePage):
# 定位元素
search_loc = (By.ID,"kw")
btn_loc = (By.ID,"su")
def open(self):
self._open(self.base_url)
def search_content(self,content):
BaiduContent = self.find_element(*self.search_loc)
BaiduContent.send_keys(content)
def btn_click(self):
BaiduBtn = self.find_element(*self.btn_loc)
BaiduBtn.click()
PageObject總結
- PO設計模式中的BasePage基類對應案例中的BasePage.py文件。
- PO模式中的page1或pageN對應案例中的Search.py
- PO設計模式中TestCase對應案例中的TestCase.py
如果你想系統學習,打個廣告吧
推廣下我的博客專欄,目前選定了一個主題《從零學Selenium自動化測試框架》,讓我們從代碼擼起,一步步實現Web自動化測試框架
該專題會從零帶你搭建一個可用的自動化測試框架(基於python+selenium)
前提:你要掌握了python與selenium基礎哦。要不你看不懂的。