web自动化测试中的PO模式(一)
1、PO模式的思想
- 原理:
- 将页面的元素定位和元素行为封装成一个page类
- 类的属性:元素的定位
- 类的行为:元素的操作
- 页面对象和测试用例分离
- 测试用例:
- 调用所需要页面对象中的行为,组成测试用例
- 测试用例中,只需要含有页面函数的调用和断言,不应该出现元素定位等其他的操作,如果写测试用例中出现需要其他的额外的操作,可以想办法封装到页面对象或者元素定位中
- 好处:
- 当某个页面的元素发生变化,只需要修改该页面对象中的代码即可,测试用例不需要修改
- 提高代码的重用率,结构清晰,维护代码容易
- 测试用例发生变化是,不需要或者只需要修改少数页面对象代码即可
2、使用unittest框架实现PO模式
- 新建一个包,命名为PageObjects,包中用来封装各个页面的功能
- 页面对象封装的类中,函数用来实现页面的功能,在类的初始化函数中,使用传参(driver)的方式完成,页面只需要实现对应功能即可,具体传入什么样的测试,在测试用例的前置条件中实现,可以提高函数的重用率
页面对象封装函数
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.by import By
class LoginPage:
def __init__(self,driver):
self.driver=driver def login(self,username,pwd,remember_user=True):
# 等待定位元素出现
WebDriverWait(self.driver,10).until(EC.visibility_of_element_located((By.XPATH,"//input[@name='phone']")))
# 输入用户名
self.driver.find_element_by_xpath("//input[@name='phone']").send_keys(username)
# 输入密码
self.driver.find_element_by_xpath("//input[@name='password']").send_keys(pwd)
# 定位记住手机号元素
remember_ele=self.driver.find_element_by_xpath("//input[@name='remember_me']")
# 判断时候记住手机号
if remember_user==True:
remember_ele.checked=True
else:
remember_ele.checked=False
# 点击登录
self.driver.find_element_by_xpath("//button[text()='登录']").click()测试用例调用页面对象函数,代码如下
1 def test_login_success(self):
2 # 2、步骤
3 # 实例化LoginPage类
4 lg=LoginPage(self.driver)
5 # 调用login方法
6 lg.login("登录账号","登录密码")
7 # 3、断言
8 self.assertTrue(IndexPage(self.driver).isExist_logout_ele())- Index_page页面封装函数代码
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.by import By
class IndexPage:
def __init__(self,driver):
self.driver=driver def isExist_logout_ele(self):
try:
WebDriverWait(self.driver,10).until(EC.visibility_of_element_located((By.XPATH,"//a[text()='退出']")))
return True
except:
return False - 测试用例的前置条件代码
import unittest
from selenium import webdriver
from PageObjects.login_page import LoginPage
from PageObjects.index_page import IndexPage
class TestLogin(unittest.TestCase): def setUp(self):
# 1、前置条件
url="http://ip:host/Index/login.html"
self.driver=webdriver.Chrome()
self.driver.maximize_window()
self.driver.get(url) - 运行测试用例可以在测试用例最后面使用一下代码,然后运行该代码
if __name__ == '__main__':
unittest.main() - 运行结果如下,表示运行成功

- 页面对象封装的类中,函数用来实现页面的功能,在类的初始化函数中,使用传参(driver)的方式完成,页面只需要实现对应功能即可,具体传入什么样的测试,在测试用例的前置条件中实现,可以提高函数的重用率
- 新建一个包,命名为TestCases,包中用来写测试用例,测试用例中用到那个页面对象中的函数时,只需要调用页面对象中的函数就可以了
- 新建一个包,命名为TestDatas,用来写测试数据,将测试数据的类导入到测试用例模块中,进行使用,这样的话,测试数据便于管理,测试数据添加、修改、删除不用修改测试用例
- 在TestCases下面新建一个公用的数据管理.py文件Common_Datas.py和模块数据的.py文件login_datas.py,如下t图:

#Common_Datas.py
web_login_url="http://ip:port/Index/login.html" #login_datas.py
#正常场景---测试数据
success_data={"user":"","pwd":""} # 异常用例--手机号格式不正确(大于11位,小于11位,为空,不在号码段)
error_data=[
{"user": "", "pwd": "","check":"请输入正确的手机号"},
{"user": "", "pwd": "","check":"请输入正确的手机号"},
{"user": "", "pwd": "","check":"请输入正确的手机号"},
{"user": "", "pwd": "","check":"请输入手机号"},
{"user": "", "pwd": "","check":"请输入密码"},
{"user": "", "pwd": "","check":"请输入手机号"},
] - 在测试用例模块使用数据管理模块中的数据
import unittest
from selenium import webdriver
from PageObjects.login_page import LoginPage
from PageObjects.index_page import IndexPage
from TestDatas import Common_Datas as CD
from TestDatas import login_datas as ld
from ddt import ddt,data @ddt
class TestLogin(unittest.TestCase):
@classmethod
def setUpClass(cls):
# 1、前置条件
cls.driver=webdriver.Chrome()
cls.driver.maximize_window()
cls.driver.get(CD.web_login_url)
# 实例化LoginPage类
cls.lg=LoginPage(cls.driver)
@classmethod
def tearDownClass(cls):
cls.driver.quit() def tearDown(self):
self.driver.refresh()
pass
# 正常用例--登录成功
def test_login_1_success(self):
# 2、步骤 # 调用login方法
self.lg.login(ld.success_data['user'],ld.success_data['pwd'])
# 3、断言
self.assertTrue(IndexPage(self.driver).isExist_logout_ele())
# 异常用例--手机号码格式不正确(大于11位,小于11位,为空,不在号码段) ddt
@data(*ld.error_data)
def test_login_0_user_wrongFormat(self,item):
self.lg.login(item['user'],item['pwd'])
self.assertEqual(self.lg.get_errorMsg_form_loginArea(),item['check'])备注:
setUpClass(cls)函数和tearDownClass(cls)只在整个用例开始和结束时运行一次,必须使用@classmethod来修饰,如上面的异常测试用例有6条,在用例开始前和结束时只运行一次
setUp(self)函数和tearDown(self)在每个测试用例开始和结束时运行,如果使用这两个函数,上面的6条用例,每运行一个,就会运行一次这两个函数,总共运行6次
tearDown(self)可以和setUpClass(cls)函数和tearDownClass(cls)函数结合使用,如上面的界面刷新self.driver.refresh()和页面关闭cls.driver.quit()
- 测试用例类型相同的测试数据可以设计在一起,使用ddt在测试用例运行时分别执行测试用例
- 使用unittest框架,运行测试用例,如下:
import unittest
from TestCases.test_login import TestLogin
from HTMLTestRunnerCN import HTMLTestReportCN
# 创建一个容器,用来存测试用例
suite=unittest.TestSuite()
# 加载测试用例的类的实例
loader=unittest.TestLoader()
# 将测试用例加载到suite容器中
suite.addTest(loader.loadTestsFromTestCase(TestLogin))
# 打开文件,用来写测试报告
with open("test_result.html",'wb') as file: # runner=unittest.TextTestRunner(verbosity=1) runner=HTMLTestReportCN( stream=file, verbosity=2, title="web自动化测试", description="第一个web自动化测试", tester="wsk")
# 运行测试用例
runner.run(suite) - 运行结果,如下图:
在写测试用例时,如果使用unittest框架可以使用函数名来控制代码的执行顺序,如:
ok test_login_0_user_wrongFormat_6 (TestCases.test_login.TestLogin) -------------------- 0
ok test_login_1_success (TestCases.test_login.TestLogin)------------------------------------------1 前面都一样,0比1小,先运行
- 在TestCases下面新建一个公用的数据管理.py文件Common_Datas.py和模块数据的.py文件login_datas.py,如下t图:
web自动化测试中的PO模式(一)的更多相关文章
- web自动化测试中绕开验证码登陆的方式
web自动化测试中登陆需验证码是很大的一个困扰.现推荐一种简单的避开验证码登陆的方式,先代码进入登录页,人工输入验证码登录后浏览器自动保存cookie,再在新的标签中登录. 具体代码如下: publi ...
- python web自动化测试中失败截图方法汇总
在使用web自动化测试中,用例失败则自动截图的网上也有,但实际能落地的却没看到,现总结在在实际应用中失败截图的几种方法: 一.使用unittest框架截图方法: 1.在tearDown中写入截图的 ...
- JavaScript在web自动化测试中的作用
前言 JS的全称JavaScript,是一种运行在浏览器中的解释型脚本语言,通常用来实现web前端页面的基本功能,对于前端开发人员是不得不掌握的一门基本技能,但是对于做web自动化测试的人员来说,如果 ...
- 说说UI自动化中的PO模式
PO模式,全称PageObject模式,即页面对象模式.将页面定位与业务操作分离. po模式有以下几个优点: 1.易读性好 2.扩展性高 3.复用性强 4.维护性好 5.代码冗余率低 了解了po模式及 ...
- TestNG测试框架在基于Selenium进行的web自动化测试中的应用
转载请注明出自天外归云的博客园:http://www.cnblogs.com/LanTianYou/ TestNG+Selenium+Ant TestNG这个测试框架可以很好的和基于Selenium的 ...
- (转)Web自动化测试中的接口测试
1.背景 1.1 Web程序中的接口 1.1.1 典型的Web设计架构 web是实现了基于网络通信的浏览器客户端与远程服务器进行交互的应用,通常包括两部分:web服务器和web客户端.web客户端的应 ...
- Web自动化测试中的接口测试
1.2.3 接口可测性分析 接口显而易见要比UI简单的都,只需要知道协议和参数即可完成一次请求,从自动化测试实施难易程度来看,有以下几个特征: 1)驱动执行接口的自动化成本不高:HTTP,RPC,SO ...
- 高访问量WEB开发中的架构模式,学习从点滴开始
当一个Web系统从日访问量10万逐步增长到1000万,甚至超过1亿的过程中,Web系统承受的压力会越来越大,在这个过程中,我们会遇到很多的问题.为了解决这些性能压力带来问题,我们需要在Web系统架构 ...
- 对web应用中单一入口模式的理解及php实现
在我们web应用的开发中,经常会听见或看见单一入口模式,在我开始学习tp框架的时候也不理解为什么要运用一个单一入口模式,只是会使用,最近自己在搞一个小东西的时候才明白为什么在web开发中要运用单一入口 ...
随机推荐
- python高阶函数&异常处理
高阶函数 1.什么是高阶函数 在Python中,变量可以指向函数 函数名也是变量 既然变量可以指向函数,函数的参数能接收变量,那么一个函数就可以接收另一个函数作为参数,这种函数就称之为高阶函数. ma ...
- JAVA开发中如何优化类的设计
具体类依赖于抽象类,而非抽象类依赖于具体类.这样做有利于一个抽象类扩展多个具体类. 开放封闭原则:对扩展开放,对修改封闭. 1.永远保持数据私有 保持数据的私有是设计类时,必须重点考虑的问题.保持私有 ...
- JDBC(三)----JDBC控制事务
## JDBC控制事务 1.事务:一个包含多个步骤的业务操作.如果这个业务员操作被事务管理,则这多个步骤要么同时成功,要么同时失败. 2.操作: 1.开启事务 2.提交事务 3.回滚事务 3.使用C ...
- 给社团同学做的R语言爬虫分享
大家好,给大家做一个关于R语言爬虫的分享,很荣幸也有些惭愧,因为我是一个编程菜鸟,社团里有很多优秀的同学经验比我要丰富的多,这次分享是很初级的,适用于没有接触过爬虫且有一些编程基础的同学,内容主要有以 ...
- OpenCV-Python 理解SVM | 五十五
目标 在这一章中 我们将对SVM有一个直观的了解 理论 线性可分数据 考虑下面的图像,它具有两种数据类型,红色和蓝色.在kNN中,对于测试数据,我们用来测量其与所有训练样本的距离,并以最小的距离作为样 ...
- GANs和低效映射
生成对抗网络(GANs)被誉为生成艺术领域的下一纪元,这是有充分理由的.新技术一直是艺术的驱动因素,从颜料的发明到照相机再到Photoshop-GAN是自然而然的.例如,考虑下面的图片,由埃尔加马勒发 ...
- Jmeter4.0接口测试之断言实战(六)
在接口测试用例中得有断言,没有断言的接口用例是无效的,一个接口的断言有三个层面,一个是HTTP状态码的断言,另外一个是业务状态码的断言,最后是某一接口请求后服务端响应数据的断言.在Jmeter中增加断 ...
- React Hooks 实现react-redux
Redux 是目前 React 系统中最常用的数据管理工具,它落实并发扬了 Flux 的数据单向流动模式,被实践证明为一种成熟可用的模式. 尽管承受着一些非议,Redux 在 React 数据管理界的 ...
- ArrayList 扩容 和 Vector
public boolean add(E e) { ensureCapacityInternal(size + 1); // Increments modCount!! elementData[siz ...
- Ubuntu在Anaconda中安装TensorFlow GPU,Keras,Pytorch
安装TensorFlow GPU pip install --ignore-installed --upgrade tensorflow-gpu 安装测试: $ source activate tf ...