一. PO模型介绍
在自动化中,
Selenium
自动化测试中有一个名字经常被提及
PageObject(
思想与面向对象的特征相
同
)
,通常
PO
模型可以大大提高测试用例的维护效率
优点:重用,业务和对象分离,代码结构清晰,方便代码维护
二
. PageObject
设计模式
三:
PO
的核心要素
1.
在
PO
模式中抽离封装集成一个
BasePage
类,该基类应该拥有一个只实现
webdriver
实例的属性
2.
每一个
page
都继承
BasePage
,通过
driver
来管理本
page
中元素,将
page
中的操作封装成一个个方法
3.TestCase
继承
unittest.Testcase
类,并依赖
page
类,从而实现相应的测试步骤
四
.
非
PO
实现
代码如下
from selenium import webdriver
from time import sleep
driver = webdriver.Firefox()
driver.get("http://www.baidu.com")
driver.find_element_by_id("kw").send_keys("jack")
sleep(2)
driver.find_element_by_id("su").click()
sleep(2)
driver.quit()
代码分析
1、不同的运行脚本环境,游览器不同:驱动 webdriver.Firefox() 可以剥离2、请求地址的变化 ( 生产环境和测试环境 ):url = http://www.baidu.com 可以剥离3、操作元素时,常常要等待元素加载完成方可进行操作:可以把 webdriver 提供的 find_element* 方法封装,在元素操作前,先判断元素是否可以操作4、实际测试场景中,可能有多个测试场景,如果每个测试场景都需要维护 url, 游览器驱动 , 元素定位等 , 效率会非常低5、因此基于以上分析,是否可以设计一个所有测试界面( selenium 本身是 B/S 系统开展测试)的基类,来 维护公共的方法,此处定义名字为BasePage.py, 用于存放页面公共方法及 webdriver 原有方法二次封装等。
五
.PO
实现
代码组织
实现
BasePage
from selenium import webdriver
from selenium.webdriver.common.action_chains import ActionChains #鼠标操作
class Basepage():
def __init__(self,driver,url):
# 实例化base类
self.driver=driver
self.base_url=url
# 进入网址
def get(self):
self.driver.get(self.base_url)
# 输入
def send_text(self,text,*locator):
self.driver.find_element(*locator).send_keys(text)
# 点击
def left_click(self,*locator):
self.driver.find_element(*locator).click()
# 清楚
def clear_text(self, *locator): self.driver.find_element(
*locator).clear()
# 表单切换
def switch_iframe(self,*locator):
self.driver.switch_to.frame(self.driver.find_element(*locator))
#窗口切换
def switch_window(self,n):
self.driver.switch_to.window(self.driver.window_handles[n])
实现
SearchPage
search-one代码如下
from selenium.webdriver.common.by import By
from basepage.basepo import Basepage
class Search(Basepage):
def __init__(self,driver,url):
Basepage.__init__(self,driver,url)
# 进入百度
def open_baidu(self):
self.get()
# 输入搜索的内容
def input_search_content(self,text):
self.send_text(text,By.ID,"kw")
# 点击按钮
def click_baidu_search(self):
self.left_click(By.ID, "su")
# 点击hao123进入
def open_123(self):
self.left_click(By.XPATH, ".//*[@id='1']/h3/a[1]")
search-two代码如下
from selenium.webdriver.common.by import By
from basepage.basepo import Basepage
class Search_two(Basepage):
def __init__(self,driver,url):
Basepage.__init__(self,driver,url)
# # 进入百度
# def open_baidu(self):
# self.get()
# # 输入搜索的内容
# def input_search_content(self,text):
# self.send_text(text,By.ID,"kw")
# # 点击按钮
# def click_baidu_search(self):
# self.left_click(By.ID, "su")
# 进入地图
def open_map(self):
self.left_click(By.XPATH,".//*[@id='userCommonSites']/ul/li[6]/div/a")
实现
TestCase
import unittest
from selenium import webdriver
from searpage.searchpo_one import Search
from searpage.searchpo_two import Search_two
class Baidu(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
cls.driver=webdriver.Firefox()
cls.driver.implicitly_wait(10)
def test_search001(self):
url="http://www.baidu.com"
s=Search(self.driver,url)
s.open_baidu()
s.input_search_content("123")
s.click_baidu_search()
s.open_123()
self.driver.switch_to.window(self.driver.window_handles[1])
def test_search002(self):
url=""
s2=Search_two(self.driver,url)
s2.open_map()
pass
@classmethod
def tearDownClass(cls) -> None:
# cls.driver.quit()
pass
if __name__ == '__main__':
unittest.main()
效果图如下:
六:
PO
模式的优点
1:PO 提供了一种业务流程与页面元素操作分离的模式,这使得测试代码变得更加清晰2 :页面对象与用例分离,使得我们更好的复用对象3 :可复用的页面方法代码会变得更加优化4 :更加有效的命令方式使得我们更加清晰的知道方法所操作的 UI 元素
七、
总结
PO 设计模式中的 BasePage 基类对应案例中的 BasePage.py 文件PO 模式中的 pages 中的案例显示 Search.pyPO 模式设计中 TestCase 对应案例中的 TestCase.py