PO(Page Object)设计模式是一种面向对象( 页面对象)的设计模式,将测试对象及单个的测试步骤封装在每个Page对象以page为单位进行管理。
优点
- 可以使代码复用
- 降低维护成本
- 提高程序可读性和编写效率。
- 可以将页面定位和业务操作分开,测试对象(元素对象)和测试脚本(用例脚本)
- 提高用例的可维护
| 非PO模式 | PO模式 |
| 面向过程的线性脚本 | POM把页面元素定位和业务操作流程分开,实现松耦合 |
| 复用性性差 | UI元素的改变不需要修改业务逻辑代码,只需要找到对应的PO页修改定位即可,数据代码分离 |
| 维护性差 | PO能使代码更具有可读性,高复用性、可维护性 |
PO三层模式:
主要分三层:
1.base层(对象库层):page页面一些公共的方法。如:初始化、元素定位、点击、输入、获取文本、截图等方法;
2.page层(操作层):封装对元素的操作。将每个涉及的元素操作单独封装一个操作方法,然后根据需求组装操作步骤,如登录方法=输入帐号+输入密码+点击登录三个操作进行组装;
3.scripts层(业务层):导包调用 page页面,使用单元测试框架对业务逻辑进行封装测试。如:实现登录,直接调用page组装的登陆方法即可。
三者的关系:page层继承base层,scripts层调用page层
案例:
项目结构介绍:

创建项目,如下图

1. 构建基础的 BasePage 对象层
创建driver,浏览器驱动封装
-
# encoding='UTF-8' -
# 浏览器启动 -
from selenium import webdriver -
def browser(): -
driver=webdriver.Chrome() -
# driver.get("http://www.baidu.com") -
return driver
创建myuni.py文件,初始化封装,
定义一个继承自unittest.TestCase的测试用例类
定义setUp和tearDown,这两个方法与junit相同,即如果定义了则会在每个测试case执行前先执行setUp方法,执行完毕后执行tearDown方法。
-
# encoding='UTF-8' -
import unittest -
from driver import * -
class StartEnd(unittest.TestCase): -
def setUp(self): -
self.driver=browser() -
self.driver.implicitly_wait(10) -
self.driver.maximize_window() -
def tearDown(self): -
self.driver.quit()
创建function.py文件,截图功能
-
# encoding='UTF-8' -
import os -
from selenium import webdriver -
# 截图 -
def insert_img(driver,filename): -
func_path=os.path.dirname(__file__) -
# print(func_path) -
base_dir=os.path.dirname(func_path) -
# print(base_dir) -
# 将路径转化为字符串 -
base_dir=str(base_dir) -
# 对路径的字符串进行替换 -
base_dir=base_dir.replace("\\","/") -
# print(base_dir) -
# 获取项目文件的要目录路径 -
base=base_dir.split('/Website')[0] -
# print(base) -
# 指定截图存放路径(注意路径最后要加/) -
filepath=base+'/Website/test_report/screnshot/'+filename -
driver.get_screenshot_as_file(filepath) -
if __name__=='__main__': -
driver=webdriver.Chrome() -
driver.get("http://www.sogou.com")
2. 构建首页的 Page 层(操作层)
创建BasePage.py文件,判断打开的页面是否是预期的页面
-
# encoding='UTF-8' -
from time import sleep -
class Page(): -
def __init__(self,driver): -
self.driver=driver -
self.base_url="http://localhost" -
self.timeout=10 -
def _open(self,url): -
url_=self.base_url+url -
# print("这个页面url是:%s"%url_) -
self.driver.maximize_window() -
self.driver.get(url_) -
# sleep(2) -
assert self.driver.current_url==url_,'这不是我们想要的地址' -
def open(self): -
self._open(self.url) -
def find_element(self,*loc): -
return self.driver.find_element(*loc)
创建LoginPage.py文件,页面元素定位封装
-
# encoding='UTF-8' -
from BasePage import * -
from selenium.webdriver.common.by import By -
class LoginPage(Page): -
url='/news/' -
username_loc=(By.NAME,'username') -
password_loc=(By.NAME,'password') -
submit_loc=(By.NAME,'Submit') -
def type_username(self,username): -
self.find_element(*self.username_loc).send_keys(username) -
def type_password(self,password): -
self.find_element(*self.password_loc).send_keys(password) -
def type_submit(self): -
self.find_element(*self.submit_loc).click() -
def login_action(self,username,password): -
self.open() -
self.type_username(username) -
self.type_password(password) -
self.type_submit() -
loginPass_loc=(By.LINK_TEXT,'我的空间') -
loginErr_loc=(By.LINK_TEXT,'加入收藏') -
def type_loginPass_hint(self): -
return self.find_element(*self.loginPass_loc).text -
def type_loginErr_hit(self): -
return self.find_element(*self.loginErr_loc).text
3.构建业务层
- 创建test_login.py文件
- 导包:function、myunit、LoginPage
- 创建LoginTest类,继承myunit.StartEnd,初始化方法
- 定义测试用例,名字以test开头,unittest会自动将test开头的方法放入测试用例集中
- 实现登录,调用LoginPage组装的登陆方法,输入用户名、密码、点击登录、断言,截图
-
# encoding='UTF-8' -
import unittest -
from model import function,myunit -
from page_object.LoginPage import * -
from time import sleep -
class LoginTest(myunit.StartEnd): -
def test_login1_normal(self): -
print("test_login1_normal测试开始") -
po=LoginPage(self.driver) -
po.login_action("yuruyi","12345678") -
sleep(5) -
self.assertEqual(po.type_loginPass_hint(),'我的空间') -
function.insert_img(self.driver,"login_normal.png") -
print("test_login1_normal执行结束") -
def test_login2_PasswdError(self): -
print("test_login2_PasswdError测试开始") -
po=LoginPage(self.driver) -
po.login_action("yuruyi","1234567") -
sleep(5) -
self.assertEqual(po.type_loginErr_hit(),'加入收藏') -
function.insert_img(self.driver,"login_Err.png") -
print("test_login2_PasswdError执行结束") -
def test_login3_empty(self): -
print("test_login3_empty测试开始") -
po = LoginPage(self.driver) -
po.login_action("", "") -
sleep(5) -
self.assertEqual(po.type_loginErr_hit(), '加入收藏') -
function.insert_img(self.driver, "login_empty.png") -
print("test_login3_empty执行结束") -
if __name__=='__main__': -
unittest.main()
常用断言方法:

4. 构建用例集,执行文件,输出自动化测试报告
在测试用例、测试文件比较多的时候,使用统一的主测试执行文件进行测试用例的执行非常方便,这就需要结合discover方法和TextTestRunner进行。
-
# encoding='UTF-8' -
import unittest -
# 测试报告模板 -
from HTMLTestRunnerCN import HTMLTestRunner -
import time -
report_dir='./test_report' -
test_dir='./test_case' -
print("start run test case") -
discover=unittest.defaultTestLoader.discover(test_dir,pattern="test_login.py") -
now=time.strftime("%Y-%m-%d %H_%M_%S") -
report_name=report_dir+'/'+now+'result.html' -
print("start write report..") -
#使用runner运行器运行测试集 -
with open(report_name,'wb')as f: -
runner=HTMLTestRunner(stream=f,title="Test Report",description="localhost login test") -
runner.run(discover) -
f.close() -
print("Test end")
测试报告模板

感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取


2019

被折叠的 条评论
为什么被折叠?



