不会做UI自动化测试?一起设计框架再实践吧

news2025/6/10 10:05:02

目的

相信做过测试的同学都听说过自动化测试,而UI自动化无论何时对测试来说都是比较吸引人的存在。

相较于接口自动化来说它可以最大程度的模拟真实用户的日常操作与特定业务场景的模拟,那么存在即合理,自动化UI测试自然也是广大测试同学职业道路上必不可少的必修课题之一了。

意义

说到UI自动化,不同的公司、不同的团队往往看待它的态度也存在着很大的差异:

  • 项目或产品是否值得做UI自动化?

  • 执行的方向是否正确?

  • 落地的成本是否过大?

大部分的测试团队都会有同样的疑问,不管初衷如何(KPI、晋升、内部推广、效率优化),最大的难点一般都在于落地后如何保持一个稳定的使用周期与实际维护的成本是否小于团队投入期望值,说人话就是用来UI自动化之后是否大家都能用且可以长久的持续与维护下去。

这里博主建议的是,在做UI自动化之前先想清楚动机是什么:

  • 如果真的只是自我拓展、KPI或者个人成果展示,那就掌握掌握原理与实操一下即可,没有必要在团队内进行推广;

  • 如果真的是解决团队的实际需求:历次回归都需大量的手工,每次右移后需要全功能回归,功能数量大、场景多、功能增量后耦合较低的情况,则可以简单的评估一下引入自动化UI测试预计带来的成果与提升效果。

设计理念

之所以选用PO模式,也正是因为一般的APP项目或产品功能都是增量式迭代开发的,那么必定会面临需要维护的功能页面越来越多的处境。

如果是传统的设计模式,页面的元素与业务的操作会全部放在一个脚本内,有点类似于面向过程的编程理念。这样的模式必定会导致编写与维护的周期与成本增加,同样也不利于团队内成员共同维护的模式。

相较于传统模式,PO(Page Object)模式则是将一个页面的所有元素对象定位和对元素对象的操作封装成类,测试用例的编写也依照单个页面来进行,目的就是实现页面对象和测试用例的分离。

这样做的好处有3点:

  • 低耦合:将每个页面单独进行封装,类似与面向对象,互相之间低耦合,即时需要业务流程连续执行也不会互相影响;

  • 易维护:当界面发生变化时,只需修改对应的页面类中的元素即可,其他相关的不会受到影响也无需修改;

  • 易上手:基于PO模式的设计理念,页面类的实现与细节不会暴露在外,都通过公共方法进行提供,使用者无需对代码的实现逻辑进行学习,只需要对业务与编程语言有足够的了解后直接编写与使用。

接下来了解了PO模式的优势之后,就需要对自动化框架进行设。

先考虑清楚使用了自动化测试框架是要解决什么问题,这里的问题不能是模糊且没有边界的,之后将要自动化的产品、模块、流程进行分类与整理,这里一般来说推荐产品的核心主流程,一般覆盖happy path即可,但如果需要加入一些反向用例与使用场景也是可以的,但切忌不要一股脑的把团队的手工测试用例都加进去,到了后期你会体验到什么叫维护的时间比测试的时间更长。

决定好以上这些了之后,就可以进行技术栈与框架的选择了,那这里我们选用的是appium+python+unittest的组合来进行PO模式测试框架的设计。当然,这里还是推荐大家根据自己的技术栈与公司环境现状来进行有效选择。

PO模式

这里先声明一点,所有的框架都不是一蹴而就的,和我们熟知的软件一样,无论是结构还是代码都是的一版一版优化出来的,所以大家现在看到的框架不会是最初与最终的模样,无论是拿来优化、二开还是直接使用都是可以的。

如果是自己写,哪怕一开始写得很简单也无所谓,要始终记住你落地自动化的目的是什么,只要能针对产品持续优化与反复总结,相信会有令人满意的结果的。

这里我们先将一个页面类分成两层,一个是对象操作层、另一个是业务层。

  • 对象操作层指的是页面中的元素定位与单个元素操作;

  • 业务层顾名思义是把对应的元素操作组合起来形成一些列的业务操作。

基于PO模式设计框架之前,我们还需要了解一下PO模式的6大原则,了解了原则之后才能更好地在实现过程中将PO模式的优势融入自己的框架之中。

6大原则

1.用公共方法代表页面提供的功能;

2.不要暴露页面元素到外部;

3.一般不在方法内加断言;

4.方法应该返回其他PO对象;

5.不需要封装页面内所有元素;

6.同样的行为不同的结果可以封装成不同的方法。

个人解读

1.一些可复用的操作,可以用公共的方法进行统一封装,即使不在同一页面;

2.封装实现方法,对外只提供方法名或接口名;

3.封装的实现方法中不要使用断言,把断言可以统一放在测试用例中;

4.可以使用其他对象作为一个方法的返回结果,比如页面的跳转,就可以用方法的结果进行返回;

5.页面中只对重要的元素进行PO设计,不重要的、非主流程的可以舍弃(这里可以更好的迎合只覆盖happy path);

6.如果一个操作可能有多种结果的时候,将结果封装成不同的方法,比如保存成功与保存失败。

框架设计

目录结构

这里简单说明下目录的结构:

  • base:存放一些框架与页面的公共方法;

  • po:存放所有的页面,这里就是被测对象相关的被测页面,不需要放全部页面;

  • result:存放相关的自动化测试结果报告;

  • test_case:存放测试用例。

根目录下还有一个run文件,这个是运行主入口,可以设置运行哪些测试用例集与使用什么样的测试报告套件。

实现步骤

这里的PO模式设计其实没有那么的复杂,从目录就可以看出,首先将一些基础的元素定位、通用操作封装到对应的BasePage类中。

这里插一句,其实做APP自动化也好,做web自动化也好,很大程度上开发的代码规范性决定了你的框架实现过程是否顺畅。所以这里大家也可以在平时的工作中与开发事先沟通好一些元素的属性写法规范,别觉得不可能,行不行事在人为。

然后根据事先整理好的业务操作流程与页面跳转关系(设计理念中提到的前置工作输出)进行功能的封装,这里推荐根据6大原则对相关操作进行实现,顺了之后就是熟练工了,大同小异的。如果日后出现了布局变更或者业务变更,统一在对应的po页面中进行修改即可。

另外,一些业务逻辑的判断,(比如是否存在该用户,不存在新建,存在直接进入),也可以放在po中,但是需要谨慎,这里比较推荐的还是放在测试用例内,也方便大家根据不同的情况做断言。

最后在页面元素、业务操作齐全的状态下进行测试用例的实现,一般来说可以先使用冒烟测试的测试用例来进行简单的业务验证,当然直接使用系统测试的测试用例也是完全没问题的,之后只需要根据之前整理好的用例选单进行转化即可。至于用例的存放目录结构可以根据po页面维度来存放,也可以根据业务维度来进行存放,见仁见智。

具体实现

  • base部分

这边先定义一个BasePage类,用来实现一些公共方法与元素定位的实现(webdriver):

classBasePage:def__init__(self, driver):self.driver = driverself.driver.implicitly_wait(10)        defby_id(self, id):returnself.driver.find_element(By.ID, id)        defby_xpath(self, xpath):returnself.driver.find_element(By.XPATH, xpath)defby_class_name(self, class_name):returnself.driver.find_element(By.CLASS_NAME, class_name)defby_uiautomator(self, uiautomator):        return self.driver.find_element(AppiumBy.ANDROID_UIAUTOMATOR, uiautomator)

(左右滑动查看完整代码)

另外后续的一些触屏的操作、元素判断也可以按需放在这里面:

defis_element(self, element):        source = self.driver.page_sourceif element insource:return Trueelse:return Falsedefdrag(self, bx=0.50, bw=0.05, by=0.4, bz=0.9):        x = self.driver.get_window_size()['width']        y = self.driver.get_window_size()['height']        sx = x * bx        ex = x * bw        sy = y * by        ey = y * bz        return self.driver.swipe(sx, sy, ex, ey, 1000)

(左右滑动查看完整代码)

这里我定义了另一个driver_setup的方法,方便每次设备启动使用:

defdriver_setup():    desired_caps = dict()    desired_caps['platformName'] = 'Android'    desired_caps['platformVersion'] = '10'    desired_caps['deviceName'] = '你自己的设备名'    desired_caps['appPackage'] = '包名'    desired_caps['appActivity'] = '启动名'    desired_caps['noReset'] = True# 不重置session信息    desired_caps['fullReset'] = False# 效果类似与卸载APP 如果不想每次重新登录,设为False    return desired_caps

(左右滑动查看完整代码)

  • po部分

目录大致如上,这里值得注意的是,不要把APP里所有的页面都加入到自动化测试中,100%的自动化测试覆盖率会让你苦不堪言,也大可不必。将每次必须回归的重要流程与高重复业务流程、场景加入即可。

以下就是po中的创建顾客页面的实现方法了,直接继承BasePage类,这里有几个例子需要关注的是,性别选择可以封装成两个方法,尽量不用同一个;另一个如果是点击类事件(单结果事件),直接click就行,不用单独在封装完元素后再进行业务操作封装,备注这样的多结果事件则要在下面单独进行业务指定。

classCustomerCreatePage(BasePage):"""    定义封装创建客户页面的各类操作    创建客    创建客户并开卡    """# 定义会员编号输入框defcustomer_number(self):returnself.by_id('com.tiffany.rta.debug:id/edt_customer_number')# 定义姓名输入框defcustomer_name(self):returnself.by_id('com.tiffany.rta.debug:id/edt_customer_name')# 定义手机输入框defcustomer_mobile(self):returnself.by_id('com.tiffany.rta.debug:id/edt_customer_mobile')# 定义性别选择defcustomer_sex(self):returnself.by_id('com.tiffany.rta.debug:id/tv_customer_sex')# 定义性别内选择项目-男defcustomer_sex_item_male(self):returnself.by_id('com.tiffany.rta.debug:id/tv_customer_boy').click()        # 定义性别内选择项目-女defcustomer_sex_item_female(self):returnself.by_id('com.tiffany.rta.debug:id/tv_customer_girl').click()        # 定义生日选择框defcustomer_birthday(self):returnself.by_id('com.tiffany.rta.debug:id/tv_customer_birthday')# 定义备注输入框defcustomer_memo(self):returnself.by_id('com.tiffany.rta.debug:id/ed_remark')# 定义保存并开卡按钮defsave_and_register_card_button(self):        return self.by_id('com.tiffany.rta.debug:id/mb_save_open_card').click()

(左右滑动查看完整代码)

接下来就是组合多个元素进行业务操作的定义:

# 定义新建顾客操作defdo_create_customer(self):self.customer_number().send_keys('00001')self.customer_name().send_keys('自动化测试01')self.customer_mobile().send_keys('13200000000')self.customer_sex()self.customer_sex_item_male()        self.save_button()

(左右滑动查看完整代码)

  • test_case部分

测试用例类继承unittest下的TestCase,初始化的时候将对应的用例业务流程加入到里面,另外在具体的测试用例中需要加对应的判断逻辑与操作步骤完整的添加在里面。

使用try捕获异常的时候记得把对应的报错名也写上,一是方便定位问题,二是有可能会导致即使用例失败,测试报告上的结果也是pass。

classTestCustomerListPage(unittest.TestCase):"""    定义客户列表界面的测试用例    创建客户    """    # 初始化必要的设备信息与业务页面defsetUp(self):self.driver = webdriver.Remote('http://localhost:4723/wd/hub', driver_setup())self.base_page = BasePage(driver=self.driver)self.home_page = HomePage(driver=self.driver)self.customer_list = CustomerListPage(driver=self.driver)self.customer_detail = CustomerDetailPage(driver=self.driver)self.customer_create = CustomerCreatePage(driver=self.driver)  # 测试用例1 -- 创建顾客deftest_1_create_customer(self):self.home_page.go_customer()        customer_name = '自动化测试01'# 业务逻辑判断 -- 是否存在该新客ifself.base_page.is_element(customer_name):self.customer_list.select_customer()self.customer_detail.do_delete_customer()self.home_page.go_index()self.home_page.go_customer()ifself.base_page.is_element(customer_name):self.customer_check.check_pass()else:self.customer_list.goto_create_customer()self.customer_create.do_create_customer()self.customer_detail.back_button()else:self.customer_list.goto_create_customer()self.customer_create.do_create_customer()self.customer_detail.back_button()try:self.assertTrue(self.driver.find_element(AppiumBy.ANDROID_UIAUTOMATOR,'new UiSelector().text("自动化测试01")'))        except NoSuchElementException as e:return e        sleep(5)deftearDown(self):self.driver.quit()if __name__ == "__main__":    unittest.main()

(左右滑动查看完整代码)

  • run部分

具体的测试用例报告模板,大家可以自由选择,这边使用的是HTMLTestReportCN,启动的方式都是大同小异的,无非就是根据自己的测试场景进行定制就行。另外测试模板的组合和样式有兴趣的同学可以自己对报告脚本进行修改,打造更适合自己团队需求的测试报告。

# 两套测试报告模板路径,只用一个的可以就定义一个report_path = os.path.join(os.getcwd() + '\\result')result_path = os.path.join(report_path, 'report.html')# 测试套件路径,根据需求修改test_dir = os.path.join(os.getcwd() + '\\test_case\\trade')# 执行指定测试用例def test_suit():    suit = unittest.TestSuite()    suit.addTest(TestOrderResultPage('test_1_order_result'))    suit.addTest(TestOrderResultPage('test_2_order_result_home_page'))return suit# 执行测试用例集dis = unittest.defaultTestLoader.discover(test_dir, pattern="test*.py")if __name__ == "__main__":with open(result_path, 'wb') as fp:        runner = HTMLTestReportCN.HTMLTestRunner(stream=fp, title='自动化APP测试报告',                                                 description='基于自动化APP测试框架产生的测试报告')        runner.run(test_suit())

(左右滑动查看完整代码)

注意点

1、PO模式虽然可以解决UI自动化测试中设计的部分问题,也仍然是目前比较主流的设计方案,后期面对大量的业务页面增加的情况,虽然可以使用通用页面来解决部分问题,但仍然避免不了界面与业务改动后大量调试代码的情况出现。

所以这也是很多公司无法将大量成本聚焦在UI自动化测试的原因,将UI自动化应用于部分主要业务的做法还是值得提倡的,它也只是提高测试团队工作效率与投入产出比的一项手段而已,千万不可本末倒置。

2、测试用例的合理设计与执行安排,如果你的测试用例的相关命名、流程设计、存放路径过于凌乱与潦草的话,相信我,后期当框架具有一定的规模后,你会发现往往在维护测试用例时花费的精力要远远大于你的执行时间。

与手工测试用例一样,无效用例始终都会出现在你的框架之中,这是无可避免的,但如何快速定位与规整这些用例就成了后期需要面对的日常问题之一,所以用例实现之初的命名规则、存放路径、实现时的备注就成了日后减少维护工作量的良好开端。

3、相较于接口自动化,UI自动化的性价比还是有一定的局限性,针对这样的情况,测试团队中如果要投入UI自动化的话可能就需要将团队中的成员定位做好一定的有效安排。

框架设计与实现的问题不大,有专业的业务理解与一定的代码功底一般都可以很好的完成对应的测试框架,这里只针对维护层面的工作来说,是专职人员定岗安排还是团队成员穿插进行都需要根据各自的团队实际情况来分配,各有利弊,毕竟维护是一件费时费力的持久性工作。


四、自动化测试学习资源分享

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

下方这份完整的软件测试视频学习教程已经上传CSDN官方认证的二维码,朋友们如果需要可以自行免费领取 【保证100%免费】

这些资料,对于想进阶【自动化测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!凡事要趁早,特别是技术行业,一定要提升技术功底。希望对大家有所帮助……基础知识、Linux必备、Shell、互联网程序原理、Mysql数据库、抓包工具专题、接口测试工具、测试进阶-Python编程、Web自动化测试、APP自动化测试、接口自动化测试、测试高级持续集成、测试架构开发测试框架、性能测试、安全测试等配套学习资源免费分享~

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

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

相关文章

Jmeter之实现参数化的不同方式详解

参数化简介 定义:动态的获取、设置或生成数据,是一种由程序驱动代替人工驱动的数据设计方案,提高脚本的编写效率以及编写质量 适用场景:当提交的数据量较大时,每次修改太麻烦,可以使用参数化 本文介绍实现…

DAMA数据管理知识体系指南之数据仓库和商务智能管理

第9章 数据仓库和商务智能管理 9.1简介 数据仓库(Data Warehouse,DW)由两个主要部分构成:首先是一个整合的决策支持数据库,其次是用于收集、清洗、转换、存储来自于各种操作型数据源和外部数据源数据的相关软件程序。两者结合以支持历史的、…

mycat学习

安装部署下载1:mycat2-install-template-1.21.zip下载2:mycat2-1.21-release-jar-with-dependencies.jar解压mycat2-install-template-1.21.zipunzip mycat2-install-template-1.21.zip把mycat2-1.21-release-jar-with-dependencies.jar放在mycat/lib中修…

【逐步剖C】-第五章-指针初阶

一、指针的基本知识 1. 预备知识: (1)内存的简单概念: 内存是电脑上特别重要的存储器,计算机中程序的运行都是在内存中进行。 为了有效的使用内存,就把内存划分成一个个小的内存单元,经过仔细…

js方法总结:join 、split、splice、async await

async await 异步里面有同步操作 执行完一个,再去执行另一个是同步,如promise().then() 所有的一起执行是异步 同步和异步操作的区别就是是否阻碍后续代码的执行。 下面例子是同步,执行完 test再去执行test2最后test3,&#xff…

Android 性能优化之内存优化与泄漏分析工具LeakCanary

一、背景 在Android应用中,除了正常的业务开发,我们也要关注性能问题。卡顿、内存溢出、内存泄漏等问题,直接的表现会反馈到用户体验上,用户体验不好导致应用被卸载或者换到其他平台。 在性能优化,各大公司都会付出一些…

sass预处理中文教程(2023开学版)

Sass安装步骤 一、Sass的安装 由于Sass是使用Ruby编写的,所以我们在使用Sass之前需要安装一下Ruby。下面是Windows平台的安装步骤。对于Mac平台,请查看这里:Mac平台安装Sass。 【步骤1“下载Ruby”】:大家到Ruby的官网&#xf…

MySQL----内置函数

MySQL---->内置函数 函数:将经常使用的代码封装起来,需要的时候直接调用就可以 从函数定义角度,函数可分为: 内置函数:系统内置的通用函数自定义函数:需要根据需求编写的函数 MySQL提供的内置函数从…

【游戏逆向】RPG游戏自动打怪之取得最近怪物

RPG游戏辅助想要自动打怪,首要的任务就是需要获得最近的怪物,否则打怪的顺序会非常混乱。 想要取得最近怪物信息 要2个步骤 第一步 计算出所有怪物的距离 第二步 对所有怪物的距离进行判断,筛选出距离最近的 第一步怪物距离计算 如果是以2D的平面来看 2个点的距…

[Tomcat下载安装以及配置(详细教程)]

下载及安装Tomcat进入tomcat官网,Tomcat官网Apache Tomcat - Welcome!选择需要下载的版本,点击下载下载路径一定要记住,并且路径中尽量不要有中文下载后是压缩包 .zip,解压后tomcat系统各个文件夹目录是什么意义:bin&a…

二进制部署K8S集群

目录 一、架构图 二、部署步骤 1、实验环境 2、操作系统初始化配置 3、部署 docker引擎 4、部署 etcd 集群 5、部署 Master 组件 一、架构图 二、部署步骤 1、实验环境 服务器类型IP地址master192.168.80.5node01192.168.80.8node02192.168.80.9 2、操作系统初始化配置…

【C++】类和对象--类的6个默认成员函数

目录1.类的6个默认成员函数2.构造函数2.1概念2.2特性3.析构函数3.1概念3.2特性4.拷贝构造函数4.1概念4.2特征5.赋值运算符重载5.1运算符重载5.2赋值运算符重载5.3前置和后置重载5.4流插入和流提取运算符重载6.const成员7.取地址重载和const取地址操作符重载1.类的6个默认成员函…

“搜索大战”正式打响,微软发布ChatGPT版搜索引擎和浏览器

微软公司宣布推出由ChatGPT支持的最新版本Bing(必应)搜索引擎和Edge浏览器,今天上线,免费使用! 自去年开始,Stable Diffusion、ChatGPT 等 AI 工具的横空出世,貌似在告诉人们“AI 正在准备重塑整…

常见的编程语言有哪些?

一、编程语言定义 编程语言的定义是指主要用于人和计算机之间通信的语言,它既能够让程序员能够准确的定义计算机所需数据,也能让计算机精准的识别人的意图。 二、编程语言排行榜以上排名信息来源于Tiobe公司公布的编程语言排行榜,近两年由于大…

【学习笔记】Nginx学习

Nginx是高性能的HTTP和反向代理的web服务器,占用内存小,处理高并发连接强,处理静态文件好耗费内存少但是不支持Java语言,Java程序只能通过与Tomcat配合完成正向代理客户端无法直接访问目标服务器,而是通过向代理服务器…

微信小程序 Springboot英语在线学习助手系统 uniapp

四六级助手系统用户端是基于微信小程序端,管理员端是基于web端,本系统是基于java编程语言,mysql数据库,idea开发工具, 系统分为用户和管理员两个角色,其中用户可以注册登陆小程序,查看英语四六级…

Java后端项目IDEA配置代码规范检查,使用checkStyle实现

最近的Java后端项目想实现代码的规范检查,调研了一圈,终于找到了简单的方式实现:以下是常见的几种方案: 1、在客户端做 git hook,主要是用 pre-commit 这个钩子。前端项目中常见的 husky 就是基于此实现的。但缺点也很…

Revit过滤器怎么用?过滤可见性操作方法

一、Revit中关于项目传递“可见性中设置的过滤器规则”的方法 设计院结构专业给机电专业提资,为了对于净高有一个直观快捷的表达,结构专业工程师就可以利用Revit可见性(快捷键:VV)中的过滤器来给本专业的梁、板相对标高用颜色进行区分&#x…

python爬虫工程师 | 都会遇到的反爬手段,详细展示低难度反爬

在爬虫实战过程中,常见的反爬手段如下所示。 IP 地址限制:网站可以检测爬虫的 IP 地址,并限制爬虫访问。User-Agent 限制:网站可以通过检测请求头中的 User-Agent 来识别爬虫。Referrer 限制:网站可以通过检测请求头中…

TCO-PNB ester,1438415-89-4 ,反式环辛烯对硝基苯酯,可用于标记蛋白质

TCO-PNB ester,TCO-PNB,反式环辛烯-对硝基苯酯 ,反式环辛烯对硝基苯酯,TCO-PNB酯产品结构式:产品规格:1.CAS号:1438415-89-4 2.分子式:C15H17NO53.分子量:291.34.包装规格…