百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

秒懂自动化测试中的PO模式

itomcoil 2025-01-10 14:21 27 浏览

自动化测试在软件项目团队中的作用举足轻重,众所周知合理地展开自动化测试,可以有效降低错误修复成本,并对软件质量保证(QA)过程起到全面推进的作用。


以web自动化为例,在Selenium框架的辅助下,自动化测试工程师们编写web自动化测试脚本似乎是一项较简单的任务,但在项目进展过程中,尤其是迭代频繁的敏捷项目研发模式下,测试人员不经意而实现的那些低质量自动化测试脚本往往会引发后期高昂的代码维护成本,久而久之所谓的自动化测试在你的团队中就形同虚设了。


回想那些年被UI自动化坑过的桥段历历在目,当遇上敏捷项目中“短频快”迭代交付时,你的自动化测试脚本很可能成为团队成本开销最高昂的任务之一。


试想更改50个测试脚本中所依赖的某web页面元素(也许就1个元素)将涉及到至少50+个自动化测试脚本的更新,更新脚本的同时不可避免会出现一些其他连锁问题(不知道用蝴蝶效应来比喻这类糟糕的情况是否合适)。这项工程不仅耗时长,而且对于早期自动化测试在团队中的推广与实践也是一个严重的阻碍因素。


都说良好的开端是成功的一半,自动化测试的落地亦是如此,在脚本实现之前,我们需要有个相对精良的设计,面对避无可避的UI元素变更,除了将那些变更频度不高的回归测试进行自动化,还有什么方式能够降低我们日后对于脚本开销带来的高昂成本呢?如何避免一个元素定位变更导致“牵一发而动全身”呢?我们是否能对变更进行限制,只动一个地方,并且让每个与之相关的测试脚本都能使用它?回答是肯定的,相信大家不难得出PO模式可以帮我们落实。


下面我们就来聊一聊Selenium中如何使用页面对象模型(PO —— Page Object)来实现可维护和可重用的自动化测试脚本,让你秒懂PO在UI自动化中的价值。

1


Selenium模拟Github自动化登录流程

这里我们先以Github (http://6tt.co/rV8V) 为例,结合selenium自动化实现模拟登录:




完整登录流程代码如下:



为了覆盖更多webdriver中的定位方式,这里以ID, NAME,CSS_SELECTOR,XPATH分别对用户名文本框,密码框,登录按钮,以及登录后页面上显示的“Start a project”按钮进行定位;



可以发现如果以上四个页面元素,但凡有一个发生变化,我们需要对该脚本进行更新,如果这些定位发生变化的元素同时还存在于其他自动化脚本文件中,我们需要依次更新每个牵连其中的脚本文件,就整个项目而言,脚本更新所花费的开销还是相当大的。

2


Selenium+unittest实现Github自动化登录测试脚本

还是以上脚本和登录场景,既然是自动化测试,我们就结合python自带的unittest框架,进行自动化测试用例脚本编写,完整代码如下:



从脚本中可见,我们通过unittest把登录业务单独封装成了一个test_login()方法,即单独的测试用例,通过assertEqual进行断言判断登录后是否出现预期的信息:



同样,如果某个控件位置发生了变化,所有用到这个控件的测试用例脚本文件都需要进行更新,耦合度越高代码维护成本越高,耗费的时间精力指数之高更是无从计量。


我们迫切需要合理的解决方案 —— PO模式来重构自动测试脚本的层次。

3


PO页面对象模型

为什么需要页面对象模型(PO)

脚本维护的主要问题是,如果100个不同的脚本使用相同的页面元素,只要对该元素做了任何更改,就需要更改所有脚本,耗时且容易出错。


一种用于脚本维护更好的方法是为每一个自动化测试所需的web页面创建一个单独的类文件,用于查找、填充或验证该页面中的web元素。任何用到该web页面元素的脚本都可以通过调用这个类。如果该web页面元素发生了变化,我们只需要在其对应的类文件中进行更改即可,而不必在100个不同的脚本中逐一更新。这种方法就是页面对象模型(POM),它有助于提高代码的可读性、可维护性和可重用性。



PO的实现

页面对象模型(PO)的基本结构:将单个页面上所有Web元素和在这些Web元素上操作的方法都放在一个Page类文件中,而用于验证的测试用例需要单独作为测试方法的一部分。


我们还是以github登录流程为例,结合页面对象模型(PO)实现自动化测试,首先我们需要将登录页面作为一个类将其封装,并单独提炼出该页面中需要用到的元素(类的成员属性),把对所需元素的操作封装成独立的方法(类的成员方法):



在Login Page类中,我们需要用到页面url, 用户名输入框,密码框,登录按钮,登陆后Button“Start a project”,所以该类共有5个成员



该类需要一个初始化方法,其他操作有:打开页面,用户名输入,密码输入,登录按钮点击,登录后“Start a project”按钮文字提取,所以Login Page类的成员方法有:



同时为了做到测试脚本与页面元素对象的分离,针对每个页面进行的测试用例脚本我们也单独存放,BaseTest脚本文件中仅仅执行通用的操作:




针对每一个功能业务流测试,单独进行测试脚本的编写:



完整版分层代码实现如下:

【PO模式下Selenium + unittest实现Github自动化登录测试脚本】



由此可见,无论login页面元素发生任何变化,我们需要修改的仅仅只是Login Page类,除此之外其余脚本均不受影响,页面对象模型大大降低了自动化脚本的维护成本及因脚本更新引发的出错概率。

4


框架的优化

在PO模式协助下,被测页面中任意元素定位发生变更都能轻而易举地解决自动化脚本的更新问题。到此是不是说明大功告成了呢?是否还存在可优化的地方呢?答案是肯定的,PO模式让我们仅需要在Page页面类单个文件中对变化的元素定位进行更新,那是否尽可能使任何一个自动化脚本文件都不发生变更呢,即便是Page页面类本身?


很简单,只需要把这些定位作为数据单独存放在配置文件中(配置文件的类型可以是csv, excel,txt),通常csv文件较为常见。


下面我们把Login Page中需要定位的元素单独放到如下csv文件中,login_page.csv文件的第一行是标题,用来对每个字段做描述,以下每一行代表一组数据,不同字段数据间隔用逗号区分(当然用空格也行,只不过csv文件默认逗号间隔):



以第二行为例,“url”见名知意,表示这行是url相关的数据,“http://6tt.co/rV8V”就是url的值;


第三行中“username_id”表示这样用来存放username的定位,通过id来定位,“login_

field”就是username的id定位值(可参见我们在login page中写的代码),下面几行是同样的含义。如此我们就把元素/定位/定位后对应的值从python脚本文件中独立出来了。


对于上面的“login_page.py”PO类文件则更改如下,通过读取csv配置文件中的内容,对login page中的成员属性进行赋值,而不必将这些属性值硬编码在python文件中,这样一来即便哪个元素的定位发生变化,我们只需更改csv配置文件即可:



更改后完整版的login_page.py源码如下:



将配置数据单独放在文件中,很容易让我们联想到可以将测试数据也从测试用例方法中剥离出来,即如果我们有多组待测账号,也可以通过同样的方式做到数据与测试脚本的分离,也就是我们经常提到的DDT数据驱动测试,这部分优化大家可以后续自行尝试,实现方式和以上实现配置文件读取方式雷同,这里就不展开了。

5


PO的优势

  • 页面对象模型提倡将单个页面元素的操作,元素间组成的业务流,以及操作后的验证,三者互相独立。这样可以使我们的代码更清晰、更容易理解;


  • 页面对象独立于测试用例,这样我们就可以根据项目需求,用不同的测试框架作用于相同的页面对象。例如,我们可以将PO与Unittest / Pytest / TestNG / Junit集成在一起进行WEB功能测试,也可以和JBehave/Cucumber集成在一起进行验收测试;


  • 由于PO类中的方法具有可重性,这就大大减少了代码冗余,代码变得更少、更优化;


  • 通过将PO类中定义的方法名更具真实意义,我们可以很容易地将UI中发生的操作与方法名映射在一起。例如,如果在点击按钮后我们到达了用户信息面,方法可以命名为'gotoUserInfo()';


100%完美设计模型不存在,PO模式也会有自己的限制,但针对web自动化不失为一个较好的解决问题方案,最后我们以思维导图总结本次分享的核心技术点:


相关推荐

python创建文件夹,轻松搞定,喝咖啡去了

最近经常在录视频课程,一个课程下面往往有许多小课,需要分多个文件夹来放视频、PPT和案例,这下可好了,一个一个手工创建,手酸了都做不完。别急,来段PYTHON代码,轻松搞定,喝咖啡去了!import...

如何编写第一个Python程序_pycharm写第一个python程序

一、第一个python程序[掌握]python:python解释器,将python代码解释成计算机认识的语言pycharm:IDE(集成开发环境),写代码的一个软件,集成了写代码,...

Python文件怎么打包为exe程序?_python3.8打包成exe文件

PyInstaller是一个Python应用程序打包工具,它可以将Python程序打包为单个独立可执行文件。要使用PyInstaller打包Python程序,需要在命令行中使用py...

官方的Python环境_python环境版本

Python是一种解释型编程开发语言,根据Python语法编写出来的程序,需要经过Python解释器来进行执行。打开Python官网(https://www.python.org),找到下载页面,选择...

[编程基础] Python配置文件读取库ConfigParser总结

PythonConfigParser教程显示了如何使用ConfigParser在Python中使用配置文件。文章目录1介绍1.1PythonConfigParser读取文件1.2Python...

Python打包exe软件,用这个库真的很容易

初学Python的人会觉得开发一个exe软件非常复杂,其实不然,从.py到.exe文件的过程很简单。你甚至可以在一天之内用Python开发一个能正常运行的exe软件,因为Python有专门exe打包库...

2025 PyInstaller 打包说明(中文指南),python 打包成exe 都在这里

点赞标记,明天就能用上这几个技巧!linux运维、shell、python、网络爬虫、数据采集等定定做,请私信。。。PyInstaller打包说明(中文指南)下面按准备→基本使用→常用...

Python自动化办公应用学习笔记40—文件路径2

4.特殊路径操作用户主目录·获取当前用户的主目录路径非常常用:frompathlibimportPathhome_dir=Path.home()#返回当前用户主目录的Path对象...

Python内置tempfile模块: 生成临时文件和目录详解

1.引言在Python开发中,临时文件和目录的创建和管理是一个常见的需求。Python提供了内置模块tempfile,用于生成临时文件和目录。本文将详细介绍tempfile模块的使用方法、原理及相关...

python代码实现读取文件并生成韦恩图

00、背景今天战略解码,有同学用韦恩图展示各个产品线的占比,效果不错。韦恩图(Venndiagram),是在集合论数学分支中,在不太严格的意义下用以表示集合的一种图解。它们用于展示在不同的事物群组之...

Python技术解放双手,一键搞定海量文件重命名,一周工作量秒搞定

摘要:想象一下,周五傍晚,办公室的同事们纷纷准备享受周末,而你,面对着堆积如山的文件,需要将它们的文件名从美国日期格式改为欧洲日期格式,这似乎注定了你将与加班为伍。但别担心,Python自动化办公来...

Python路径操作的一些基础方法_python路径文件

带你走进@机器人时代Discover点击上面蓝色文字,关注我们Python自动化操作文件避开不了路径操作方法,今天我们来学习一下路径操作的一些基础。Pathlib库模块提供的路径操作包括路径的...

Python爬取下载m3u8加密视频,原来这么简单

1.前言爬取视频的时候发现,现在的视频都是经过加密(m3u8),不再是mp4或者avi链接直接在网页显示,都是经过加密形成ts文件分段进行播放。今天就教大家如果通过python爬取下载m3u8加密视频...

探秘 shutil:Python 高级文件操作的得力助手

在Python的标准库中,shutil模块犹如一位技艺精湛的工匠,为我们处理文件和目录提供了一系列高级操作功能。无论是文件的复制、移动、删除,还是归档与解压缩,shutil都能以简洁高效的方式完成...

怎么把 Python + Flet 开发的程序,打包为 exe ?这个方法很简单!

前面用Python+Flet开发的“我的计算器v3”,怎么打包为exe文件呢?这样才能分发给他人,直接“双击”运行使用啊!今天我给大家分享一个简单的、可用的,把Flet开发的程序打包为...