2023面试必备:web自动化测试POM设计模式详解

news2025/7/27 17:43:38

1.背景

为UI页面写自动化测试用例时(如:web自动化、app自动化),使用普通的线性代码,测试用例中会存在大量的元素定位及操作细节,当UI界面变化时,测试用例也要跟着变化,在自动化测试用例增加时,用例将难以维护。那么有没有一个方法能够降低UI自动化测试用例的维护难度呢?PageObject设计模式(简称PO模式)很好的解决了这个问题

2.PO简介

PO是PageObject设计模式的简称,有时也称为POM,中文含义:页面对象模型,用于专门对一个系统进行自动化测试实现的设计模式。可以最大程度实现测试的覆盖率,是目前业内公认的最佳设计模式。

3.理解PO模式

我们知道做UI自动化时,就是使用自动化的方式对页面元素进行相关的操作,业务流程就是通过对不同UI页面进行操作、跳转,从而构成我们的主体业务流程。PO模式就是以每个页面为对象主体作为自动化测试的执行对象,来实现整体的自动化测试。下面我们以添加商品到用户购物车为例进行普通思路与PO模式的对比帮助大家进一步理解PO模式

添加商品到购物车的测试

---普通模式

输入url-->点击登录-->输入账号-->输入密码
-->点击登录按钮-->搜索商品-->选择商品-->
选择商品属性-->点击加入购物车

---PO模式

进入登录页,执行登录操作

进入商品详情页,执行添加商品到购物车操作

进入购物车页面,执行校验添加是否成功操作

从上面的对比,我们发现PO模式基于系统的模块页面定义不同的页面对象,来实现不同页面的一个或者多个操作行为;再基于多个页面对象的操作行为进行组装,实现一个完整的系统业务流程

4.如何实现

4.1 PO模式的工程结构

PO模式在实现时,基本上可以分为四层:基类、页面对象类、测试数据类、测试用例类;通过分层架构的思想,实现测试用例与测试数据的分离、业务代码与测试代码的分离,具体每层的作用如下:

基类(base_page):封装各类行为操作,便于测试页面对象类进行调用,是整个PO模式体系的底层实现

页面对象类(obj_page):提取系统中的关键页面,封装成页面对象;主要页面的元素封装为页面对象类的属性,将页面的操作行为封装为页面对象类的方法

测试数据类(case_data):管理测试数据

测试用例类(test_cases):组装各页面对象类的行为,形成完成的业务流程进行测试

具体的工程图例如下:

 

4.2 工程代码

下面以ecshop项目为例演绎整体的框架设计思路

a.基类的包下创建一个basepage.py文件,里面封装底层的基础操作及公共方法,详细代码如下:

from selenium import webdriver
from time import sleep

class BasePage:
    #初始化
    def __init__(self,brow_type):
        try:
            driver = getattr(webdriver, brow_type)()
        except:
            driver = getattr(webdriver, 'Chrome')()
        driver.maximize_window()
        self.driver = driver

    #封装get方法
    def get(self,url):
        self.driver.get(url)

    #封装定位页面元素的方法
    def locator(self,locator):
        return self.driver.find_element(*locator)

    #封装input方法
    def input(self,locator,input_value):
        self.locator(locator).send_keys(input_value)

    #封装close方法
    def close(self):
        self.driver.close()

    #封装quit方法
    def quit(self):
        self.driver.quit()

    #封装click方法
    def click(self,locator):
        self.locator(locator).click()
    
    #封装强制等待方法
    def sleep(self,sec):
        sleep(sec)

b.页面对象类的包下为每个页面对象创建一个页面对象的.py文件,这里我们举例创建一个login_page.py文件,详细代码如下:

from base_page.basepage import BasePage

class LoginPage(BasePage):
    '''
    页面对象类继承于基类
    页面元素类对象
    1.页面url:封装为页面对象类的属性
    2.页面元素:封装为页面对象类的属性
    3.页面功能:封装页面对象类的方法
    '''
    #1.页面url
    default_url = r'http://192.168.53.213/ecshop'
    login_url = r'/user.php'
    url = default_url + login_url
    #2.页面元素
    username = ('name','username')
    password = ('name','password')
    click_locator = ('name','submit')
    #3.页面功能
    def login(self,user,pwd):
        self.get(self.url)
        self.input(self.username,user)
        self.input(self.password,pwd)
        self.click(self.click_locator)

c.在测试数据类的包下面使用yaml格式数据管理测试用例数据,数据具体内容如下:

d.测试用例类的包下创建test_开头的测试用例.py文件,这里我们举例创建一个test_login.py文件,详细代码如下:

**import unittest
from ddt import ddt,file_data
from obj_page.login_page import LoginPage

@ddt
class TestLogin(unittest.TestCase):
    #前置处理:创建登录页面对象
    def setUp(self) -> None:
        self.loginpage = LoginPage('Chrome')
        
    #后置处理:退出浏览器
    def tearDown(self) -> None:
        self.loginpage.quit()
    
    #测试登录,结合ddt+yaml进行数据驱动处理   
    @file_data('../case_data/login.yaml')
    def test_login(self,user,pwd,locator,except_value):
        self.loginpage.login(user,pwd)#登录
        self.assertEqual(self.loginpage.locator(eval(locator)).text,except_value)#断言

if __name__ == '__main__':
    unittest.main()

4.3 执行测试用例查看结果

4.4 这样设计具体优点

a.当登录页面的页面元素位置变化时,我们只需要在页面对象类中去具体修改具体页面对应的元素即可,不需要去修改测试用例的内容,如下图:

b.当登录的业务流程变化时,比如要加上勾选协议条款等必选操作时,我们只需要在页面对象类中去修改具体对应的操作行为即可,不需求做其它任何修改,如下图:

c.我们要测试登录的多种场景时,只需要去测试用例数据管理类中找到对应的数据管理文档,修改或添加测试用例数据即可,不需求做其它任何操作,大大的提高了代码的复用性,如下图:

d.对于相互依赖的复杂业务场景,当业务流程变化时,我们只需要修改测试用例对于各页面对象类行为方法的组装过程,受流程影响的测试数据同步修改即可,不需求修改其它的地方。如下图:

5.PO模式设计总结

a.公共方法代表页面提供的服务

b.不要暴露页面细节,对外只提供方法名

c.不要把断言和操作细节混用,操作细节中不要使用断言,可以把断言放到单独的模块中,如:testcase中

d.页面对象类的方法可以return到新打开的页面,点击一个页面可能会产生新的页面,可以使用return表示跳转,或者通过页面url直接访问

e.不要把整页内容都放到PO中,只为页面中重要的内容进行封装,对不重要的内容进行舍弃

f.相同的行为会产生不同的结果,可以封装不同的接口;如点击登录可能成功,也可能失败,你可以封装为不同的方法;也可以借用测试用例数据来处理

g.UI自动测试以流程测试为主,不是为了所以的细节全部实现自动化

  如果文章对你有帮助,记得点赞,收藏,加关注。会不定期分享一些干货哦......

最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于想做【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!凡事要趁早,特别是技术行业,一定要提升技术功底。希望对大家有所帮助……加入我的学习交流群一起学习交流讨论把!!!!

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/368210.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

初学C/C++内存管理--new和delete的使用

一,内存分布 栈区: 一般的局部变量和函数的返回数据以及返回地址,函数的参数都在战栈区上开辟空间。栈区开空间一般由编译器自动管理,出了生命周期自动释放。也可以通过一些方式自己手动开辟栈区空间,不过一般用不到…

代码随想录【Day23】| 669. 修剪二叉搜索树、108. 将有序数组转换为二叉搜索树、538. 把二叉搜索树转换为累加树

669. 修剪二叉搜索树 题目链接 题目描述: 给定一个二叉搜索树,同时给定最小边界L 和最大边界 R。通过修剪二叉搜索树,使得所有节点的值在[L, R]中 (R>L) 。你可能需要改变树的根节点,所以结果应当返回修剪好的二叉搜索树的新…

【python学习笔记】:如何去除字符串中空格

在输入数据时,很有可能会无意中输入多余的空格,或者在一些场景中,字符串前后不允许出现空格和特殊字符,此时就需要去除字符串中的空格和特殊字符。 【注意:这里的特殊字符,指的是制表符(\t&…

2023/2/24 图数据库Neo4j的理解与应用

1 什么是图数据库(graph database) 十大应用案例:https://go.neo4j.com/rs/710-RRC-335/images/Neo4j-Top-Use-Cases-ZH.pdf “大数据”每年都在增长,但如今的企业领导者不仅需要管理更大规模的数据,还迫切需要从现有…

13、Swin Transformer: Hierarchical Vision Transformer using Shifted Windows

简介 主页:https://github. com/microsoft/Swin-Transformer. Swin Transformer 是 2021 ICCV最佳论文,屠榜了各大CV任务,性能优于DeiT、ViT和EfficientNet等主干网络,已经替代经典的CNN架构,成为了计算机视觉领域通用…

第十三届蓝桥杯省赛 C++ C 组 I 题、Python B 组 H题——技能升级(AC)

目录1.技能升级1.题目描述2.输入格式3.输出格式4.样例输入5.样例输出6.数据范围7.原题链接2.解题思路3Ac_code1.技能升级 1.题目描述 小蓝最近正在玩一款 RPG 游戏。 他的角色一共有 NNN 个可以加攻击力的技能。 其中第 iii个技能首次升级可以提升 AiA_iAi​点攻击力&#x…

django项目实战十(django+bootstrap实现增删改查)进阶数据统计

目录 一、echarts 1、下载 2、配置 二、实现统计分析页面--架构和柱图 1、url 2、chart.py 3、chart_list.html 4、修改url 5、新增chart_bar方法 6、修改chart_list.html 四、饼图 1、url 2、视图chart.py新增 3、修改chart_list.html 五、折线图 1、url 2、char…

Redis在windows本地的安装配置

Redis在windows本地的安装配置 一、安装redis 1. Redis官网下载地址:http://redis.io/download,下载相应版本的Redis,在运行中输入cmd,然后把目录指向解压的Redis目录。 2、启动服务命令 输入命令:redis-server red…

数据结构(六):冒泡排序、选择排序、插入排序、希尔排序、快速排序

数据结构(六)一、大O表示法二、冒泡排序三、选择排序一、大O表示法 在计算机中采用粗略的度量来描述计算机算法的效率,这种方法被称为“大O”表示法。 我们判断一个算法的效率,不能只凭着算法运行的速度,因为随着数据…

为什么都在喊数据可视化?它究竟怎么做?

在数字化转型的浪潮中,不论是传统行业,还是新兴行业总会提到“数据可视化”这个词。那数据可视化到底是什么?为什么会受到那么多人追捧?又该怎么才能做到数据可视化呢? 一、数据可视化是什么? 首先“可视…

VC++ 解决dll库动态库加载失败问题(调用LoadLibrary加载失败)(附源码)

目录 1、动态加载dll库去调用库中的函数 1.1、调用系统dll库中未公开的接口 1.2、调用控件库中的注册接口向系统中注册该控件 2、LoadLibrary动态加载dll库失败的场景 2.1、自制安装包中遇到的LoadLibrary加载dll库失败问题 2.2、主程序底层模块调用LoadLibrary加载dll库…

企企通聚源池| 聚合海量资源全网寻源,赋能供采双方撮合交易

目前,我们正处于一个飞速发展的信息时代,随着大数据时代的来临,在企业的日常经营中,数据无处不在,各类数据的采集、整合、分析对企业的发展、决策有着十分重要的作用。数据管理作为企业一项重要的建设工作,…

1089. 烽火传递

Powered by:NEFU AB-IN Link 文章目录1089. 烽火传递题意思路代码1089. 烽火传递 题意 烽火台是重要的军事防御设施,一般建在交通要道或险要处。 一旦有军情发生,则白天用浓烟,晚上有火光传递军情。 在某两个城市之间有 n座烽火台&#xff0…

ChatGPT也懂如何设计开发板!?

到底应该如何设计一款开发板?我们问了一下最近风很大的ChatGPT,得出了这样的回答: 或者这样的回答: 显而易见,RK3568开发板是一款功能丰富,性能优异,易于开发的高性能开发板,适用于各…

SpringIOC源码解析

Spring深度学习(一)——IOC的设计理念Spring的核心思想——IOCSpring流程图DEMO编写Spring IoC容器的加载过程实例化化容器:AnnotationConfigApplicationContext实例化建BeanDefinition读取器: AnnotatedBeanDefinitionReaderBean…

设计模式之建造者模式(C++)

作者:翟天保Steven 版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处 一、建造者模式是什么? 建造者模式是一种创建型的软件设计模式,用于构造相对复杂的对象。 建造者模式可以…

【GANs】什么是饱和损失函数 Non-Saturating LossFunction

Saturating VS Non-Saturating Loss functions in GANs【GANs】什么是饱和损失函数 Non-Saturating LossFunctionSaturating VS Non-Saturating Loss functions in GANs 饱和Loss 普通GAN loss是生成器希望最小化被判断为假的概率。x取值范围是[0,1],所以图中函数…

一篇五分生信临床模型预测文章代码复现——FIgure 9.列线图构建,ROC分析,DCA分析 (五)

之前讲过临床模型预测的专栏,但那只是基础版本,下面我们以自噬相关基因为例子,模仿一篇五分文章,将图和代码复现出来,学会本专栏课程,可以具备发一篇五分左右文章的水平: 本专栏目录如下: Figure 1:差异表达基因及预后基因筛选(图片仅供参考) Figure 2. 生存分析,…

酷开系统AI人工智能技术,为营销抢夺更多目标消费者

随着越来越多的年轻群体回归家庭,互联网电视产业正在时代的浪潮下快速发展,如今已经有数以万计的家庭消费者倾向于在客厅场景中使用大屏电视观看更多丰富的电视节目,而这一趋势,对于急需线上互动营销渠道的企业和品牌方来说&#…

论文阅读笔记——《室内服务机器人的实时场景分割算法》

一、主要工作 通过深度可分离卷积、膨胀卷积和通道注意力机制设计轻量级的高准确度特征提取模块。融合浅层特征与深层语义特征获得更丰富的图像特征。在NYUDv2和CamVid数据集上的MIoU分别达到72.7%和59.9%,模型的计算力为4.2GFLOPs,参数量为8.3Mb。 二…