一、PO模型

1、PO介绍:page(页面) object(对象)

在自动化中,Selenium 自动化测试中有一个名字经常被提及 PageObject (思想与面向对象的特征相同),通常PO 模型可以大大提高测试用例的维护效率。

优点:

提交测试脚本可读性

减少代码重复

提高测试用例的可维护性,特别是针对UI变动频繁的项目

缺点:

结构复杂:基于流程进行了模块化的拆分

结构:

  1. base(基类)
  2. page(页面对象)
  3. scripts(业务层)

扩展:

loc 变量:类型为元组:*loc为解包

2、PageObject 设计模式

3、PO 的核心要素

  1. 在 PO 模式中抽离封装集成一个 BasePage 类,该基类应该拥有一个只实现 webdriver 实例的属性

  2. 每一个page 都继承BasePage,通过 driver 来管理 page 中元素,将 page 中的操作封装成一个个方法

  3. TestCase 继承 unittest.TestCase类,并依赖page类,从而实现相应的测试步骤

二、将Selenium代码封装成PO模型

1、案例说明(简单的登录测试用例)

(1)改造案例思路:

  • 第一, 我们要分离测试对象(元素对象)和测试脚本(用例脚本),那么我们分别创建两个脚本文件,分别为:

    • LoginPage.py 用于定义页面元素对象,每一个元素都封装成组件(可以看做存放页面元素对象的仓库)
    • TestCaseLogin.py 测试用例脚本。
  • 第二,抽取出公共方法定义在base.py文件中,每个Page类都要继承这个base.py文件,也就是每Page类都能使用base类中的方法,来操作页面中的元素,同时也可以在每个Page类中定义自己独有的方法,解决工作中的实际需求。
  • 第三,设计实现思想,一切元素和元素的操作组件化定义在Page页面,用例脚本页面,通过调用Page中的组件对象,进行拼凑成一个登录脚本。

(2)封装公共操作在base类(base.py)

把一些公共的方法放到此类中,这个类将被PO对象继承。

from selenium.common.exceptions import NoSuchElementException
from selenium.webdriver import ActionChains
from selenium.webdriver.support.wait import WebDriverWait class Base(object): def __init__(self, driver):
self.driver = driver # 查找元素方法(提供:点击、输入、获取文本)使用
def base_find_element(self, loc, timeout=30, poll=0.5):
element, index = loc[:2], loc[-1]
try:
ele = WebDriverWait(
self.driver, timeout=timeout, poll_frequency=poll
).until(
lambda x: x.find_elements(*element)
)[index]
except (NoSuchElementException, TimeoutError, IndexError):
return False
else:
return ele # 点击方法
def base_click(self, loc):
self.base_find_element(loc).click() # 输入方法
def base_input(self, loc, value):
el = self.base_find_element(loc)
# 清空
el.clear()
# 输入
el.send_keys(value) # 获取文本方法
def base_get_text(self, loc):
return self.base_find_element(loc).text # 获取当前页面地址
def base_get_current_url(self):
return self.driver.current_url # 鼠标悬停事件
def base_move_to_element(self, loc):
ActionChains(self.driver).move_to_element(loc).perform()

(3)每个页面对应一个Page类(login_page.py)

定位元素的定位器和操作元素方法分离开,元素定位器全部放一起,然后每一个操作元素动作写成一个方法。

from selenium.webdriver.common.by import By
from public.base import Base login_username = (By.CLASS_NAME, 'el-input__inner', 1)
login_password = (By.CLASS_NAME, 'el-input__inner', 2)
login_btn = (By.CLASS_NAME, 'login__button', 0)
organ_name = (By.CSS_SELECTOR, '.role-names-button', 0)
logout_btn = (By.CLASS_NAME, 'logout', 0)
password_login_text = (By.CSS_SELECTOR, '.pwd-login', 0) class LoginPage(object): def __init__(self, driver):
self.element = Base(driver) # 输入手机号
def input_username(self, content):
self.element.base_input(login_username, content) # 输入密码
def input_password(self, content):
self.element.base_input(login_password, content) # 点击登录
def click_login_button(self):
self.element.base_click(login_btn) # 获取当前页面url
def get_current_url(self):
return self.element.base_get_current_url() # 点击机构名称
def click_to_organ_name(self):
self.element.base_click(organ_name) # 点击退出
def click_logout(self):
self.element.base_click(logout_btn) # 获取“密码登录”文字
def get_password_login_text(self):
return self.element.base_get_text(password_login_text)

(4)封装业务层(login_business.py)

import time

from Test.Page.login_page import *

class LoginBusiness(object):

    def __init__(self, driver):
self.login_business = LoginPage(driver) # 登录系统
def go_system(self, username, password):
self.login_business.input_username(username)
self.login_business.input_password(password)
self.login_business.click_login_button()
time.sleep(2)
return self.login_business.get_current_url() # 退出成功
def logout(self):
self.login_business.click_to_organ_name()
self.login_business.click_logout()
return self.login_business.get_password_login_text()

(4)原登陆案例封装完成代码

测试方法及测试类的执行都在此文件中。

from Test.Case.base_case import LoginBaseCase
from public.get_log import LogInfo
from public.read_ini import get_value
from Test.Business.login_business import LoginBusiness class TestLogin(LoginBaseCase, LogInfo):
""" 登录退出测试用例 """ @classmethod
def setUpClass(cls) -> None:
cls.case_forward()
LogInfo().log.info('TestLogin Cases Suite Start Running')
cls.login = LoginBusiness(cls.driver) @LogInfo.get_error
def test_1(self):
""" 登录流程 """
self.log.info('TestCase1 Start Running')
# 获取用户名密码
username = get_value(get_value('Base', 'Env'), 'username')
password = get_value(get_value('Base', 'Env'), 'password')
url = self.login.go_system(username, password)
text = "https://gssdev.haoshengy.com/pc_workbench/workbench/overview"
self.assertEqual(url, text, '当前页面URL不正确--测试不通过') def test_2(self):
""" 退出登录 """
self.log.info('TestCase2 Start Running')
text = self.login.logout()
self.assertEqual("密码登录", text, '首页密码登录字样错误--测试不通过')

三、数据驱动

1.1 什么是数据驱动

数据驱动:是以数据来驱动整个测试用例的执行,也就是测试数据决定测试结果

1.2 数据驱动的特点
  1. 数据驱动本身不是一个工业级标砖的概念,因此在不同的公司都会有不同的解释
  2. 可以把数据驱动理解成一种模式或者一种思想
  3. 数据驱动技术可以让用户把关注点放在测试数据的构建和维护上,而不是直接维护脚本,可以利用同样的过程对不同的数据输入进行测试
  4. 数据驱动的实现要依赖参数化的技术
1.3 传入数据的方式(测试数据的来源)
  1. 直接定义在测试脚本中(简单直观,但代码和数据未实现真正的分离,不方便后期维护)
  2. 从文件读取数据,如json、Excel、xml、txt、等格式文件
  3. 从数据库中读取数据
  4. 直接调用接口获取数据源
  5. 本地封装一些生成数据的方法

UI自动化学习笔记- PO模型介绍和使用的更多相关文章

  1. UI自动化学习笔记- UnitTest单元测试框架详解

    一.UnitTest基本使用 1. UnitTest框架 1.1 什么是框架 说明: 框架英文单词frame 为解决一类事情的功能集合 1.2什么是UnitTest框架 概念:UnitTest是pyt ...

  2. UI自动化学习笔记- Selenium一些特殊操作的处理

    一.特殊操作处理 1. 下拉选择框操作 1.1 如何操作下拉选择框 实现方式一 思路:先定位到要操作的option元素,然后执行点击操作 driver.find_element_by_css_sele ...

  3. UI自动化学习笔记- Selenium元素定位及元素操作

    一.元素定位 1. 如何进行元素定位? 元素定位就是通过元素的信息或元素层级结构来定位元素的 2.定位工具 浏览器开发者工具 3.元素定位方式 Selenium提供了八种定位元素方式 id name ...

  4. UI自动化学习笔记- Selenium元素等待(强制等待、显示等待、隐式等待)

    一.元素等待 1. 元素等待 1.1 什么是元素等待 概念:在定位页面元素时如果未找到,会在指定时间内一直等待的过程 意思就是:等待指定元素已被加载出来之后,我们才去定位该元素,就不会出现定位失败的现 ...

  5. UI自动化学习笔记- 日志相关操作

    一.日志相关 1.日志 概念:日志就是用于记录系统运行时的信息,对一个事件的记录,也称log 1.1 日志的作用 调试程序 了解系统程序运行的情况,是否正常 系统程序运行故障分析与问题定位 用来做用户 ...

  6. Android自动化学习笔记之MonkeyRunner:官方介绍和简单实例

    ---------------------------------------------------------------------------------------------------- ...

  7. Android自动化学习笔记:编写MonkeyRunner脚本的几种方式

    ---------------------------------------------------------------------------------------------------- ...

  8. ArcGIS案例学习笔记2_2_模型构建器和山顶点提取批处理

    ArcGIS案例学习笔记2_2_模型构建器和山顶点提取批处理 计划时间:第二天下午 背景:数据量大,工程大 目的:自动化,批处理,定制业务流程,不写程序 教程:Pdf/343 数据:chap8/ex5 ...

  9. UI设计学习笔记(7-12)

    UI学习笔记(7)--扁平化图标 认识扁平化 Flat Design 抛弃传统的渐变.阴影.高光等拟真视觉效果,打造看上去更平的界面.(颜色.形状) 扁平化图标有什么优缺点 优点: 简约不简单.有新鲜 ...

随机推荐

  1. 狂神说redis笔记(一)

    一.Nosql概述 1.单机Mysql时代 90年代,一个网站的访问量一般不会太大,单个数据库完全够用.随着用户增多,网站出现以下问题: 数据量增加到一定程度,单机数据库就放不下了 数据的索引(B+ ...

  2. 菜鸟刷题路:剑指 Offer 06. 从尾到头打印链表

    剑指 Offer 06. 从尾到头打印链表 class Solution { public int[] reversePrint(ListNode head) { Stack<Integer&g ...

  3. 利用 Bean Validation 来简化接口请求参数校验

    团队新来了个校招实习生静静,相互交流后发现竟然是我母校同实验室的小学妹,小学妹很热情地认下了我这个失散多年的大湿哥,后来... 小学妹:大湿哥,咱们项目里的 Controller 怎么都看不到参数校验 ...

  4. 【模拟8.10】Weed(线段树)

    考试只好随便骗骗分过去啦啦啦..... 正解是玄学线段树: 以每个操作为叶子节点,我们定义几个变量ce表示层数,h表示高度,add表示所减的层数 那么问题转化为单点修改的问题输出直接是根节点答案 但是 ...

  5. noip2015 总结

    神奇的幻方 题目描述 幻方是一种很神奇的N*N矩阵:它由数字1,2,3,--,N构成,且每行.每列及两条对角线上的数字之和都相同. 当N为奇数时,我们可以通过以下方法构建一个幻方:首先将1写在第一行的 ...

  6. 一个排序引发的BUG

    你好呀,我是why. 前两天在 Git 上闲逛的时候又不知不觉逛到 Dubbo 那里去了. 看了一下最近一个月的数据,社区活跃度还是很高的: 然后看了一下最新的 issue,大家提问都很积极. 其中看 ...

  7. 玩转STM32MP157- 使用fbtft驱动 lcd st7735r

    什么是fbtft fbtft 在 github 中的介绍是" Linux Framebuffer drivers for small TFT LCD display modules,翻译过来 ...

  8. .Net Core 3.1简单搭建微服务

    学如逆水行舟,不进则退!最近发现微服务真的是大势所趋,停留在公司所用框架里已经严重满足不了未来的项目需要了,所以抽空了解了一下微服务,并进行了代码落地. 虽然项目简单,但过程中确实也学到了不少东西. ...

  9. excel VBA一个fuction同时执行多个正则表达式,实现方法

    代码: Function zhengze3(ze1 As String, ze2 As String, Rng1 As Range, Rng2 As Range)    Set regx1 = Cre ...

  10. 即时通信之 SignalR

    即时通信在日常的web开发场景中经常使用,本篇主要回顾一下SignalR的实现原理和通过例子说明如何在.NET Core 中使用. SingnalR 应用 需要从服务器进行高频更新的应用程序.例如游戏 ...