从装饰器出发,优雅处理 UI 自动化中的异常
二、装饰器核心原理:函数式编程的“语法糖”
在 UI 自动化测试的实际场景中,页面加载超时、元素定位失败、网络波动等异常问题频繁出现。传统的try-except嵌套方式虽然能捕获异常,但会导致业务代码与处理逻辑高度耦合,造成代码冗余、可读性差等问题。
装饰器作为 Python 的核心特性之一,能够以非侵入式的方式将异常处理逻辑从业务代码中剥离,实现代码解耦与复用。通过装饰器,我们可以将通用的异常处理逻辑封装为可复用的组件,让测试代码更简洁、易维护。
装饰器是Python中函数式编程的重要特性,本质是通过高阶函数实现代码增强,遵循以下核心原则:
- 不修改原函数代码:通过嵌套函数包裹原逻辑
- 透明性:调用者无需感知装饰逻辑
- 复用性:一个装饰器可作用于多个函数
1. 两层装饰器(无参数版)
结构解析:
def decorator(func): # 第一层:接收被装饰函数def wrapper(*args, **kwargs): # 第二层:包装函数(命名为wrapper)# 前置逻辑(如异常处理)result = func(*args, **kwargs) # 执行原函数# 后置逻辑(如返回值处理)return resultreturn wrapper # 返回包装函数
执行流程:
@decorator # 等价于 func = decorator(func)
def original_func(): pass
2. 三层装饰器(带参数版)
结构解析:
def decorator_with_param(param): # 第一层:接收装饰器参数def decorator(func): # 第二层:接收被装饰函数def wrapper(*args, **kwargs): # 第三层:包装函数# 使用装饰器参数(param)和原函数参数(*args)result = func(*args, **kwargs)return resultreturn wrapperreturn decorator # 返回第二层函数
执行流程:
@decorator_with_param("参数值") # 等价于 func = decorator_with_param("参数值")(func)
def original_func(): pass
关键区别对比
类型 | 层数 | 参数传递时机 | 典型场景 |
---|---|---|---|
无参数装饰器 | 2层 | 装饰器定义时 | 统一日志记录 |
带参数装饰器 | 3层 | 装饰器调用时 | 多场景配置(如重试次数) |
二、装饰器在异常处理中的应用:替代try/catch的优雅方案
在UI自动化中,传统try/catch
会导致测试用例臃肿,通过装饰器可将异常处理逻辑移至框架层,实现:
- 关注点分离:测试用例仅关注业务逻辑
- 集中管理:异常处理逻辑统一维护
- 可配置性:通过参数动态调整处理策略
1. 基础异常处理装饰器(两层结构)
from selenium.common.exceptions import WebDriverException
import logginglogger = logging.getLogger("UI_Handler")def handle_ui_exception(func):def wrapper(*args, **kwargs):try:return func(*args, **kwargs) # 执行原操作except WebDriverException as e: # 捕获UI技术异常logger.error(f"UI操作失败:{str(e)}")raise # 抛给上层处理或添加重试except Exception as e:logger.error(f"未知异常:{str(e)}")raisereturn wrapper
应用在Page Object层(示例:处理元素不存在异常):
from selenium import webdriver
from selenium.webdriver.common.by import Byclass LoginPage:def __init__(self, driver):self.driver = driver@handle_ui_exception # 装饰器应用于操作方法def click_submit(self):# 可能抛出NoSuchElementExceptionself.driver.find_element(By.ID, "submit").click()
测试用例层无感知调用:
def test_login(driver):login_page = LoginPage(driver)login_page.click_submit() # 异常处理在中间层完成assert DashboardPage(driver).is_displayed()
2. 参数化异常处理装饰器(三层结构)
需求:不同场景下异常处理策略不同(如重试次数、是否截图)
示例:处理NoSuchElementException并应对业务空值:
from selenium.common.exceptions import NoSuchElementException
from functools import wraps
import logging
import timelogger = logging.getLogger("UI_Handler")def handle_element_exceptions(retry_times=3, default_value="N/A" # 业务空值默认返回值
):"""处理元素定位异常和业务空值的装饰器"""def decorator(func):@wraps(func)def wrapper(*args, **kwargs):self = args[0] # 假设第一个参数是Page Object实例driver = self.driver# 技术异常处理:重试机制for attempt in range(retry_times + 1):try:result = func(*args, **kwargs)# 业务空值处理:文本为空时返回默认值if isinstance(result, str) and len(result.strip()) == 0:logger.warning("检测到业务空值,返回默认值")return default_valuereturn resultexcept NoSuchElementException as e: # 捕获元素不存在异常if attempt < retry_times:logger.error(f"元素未找到(重试{attempt+1}/{retry_times}):{str(e)}")time.sleep(1) # 重试间隔1秒else:logger.error(f"最终失败:{str(e)}")return default_value # 超过重试次数返回默认值except Exception as e:logger.error(f"未知异常:{str(e)}")return default_valuereturn wrapperreturn decorator
Page Object层应用示例:
class SearchPage:def __init__(self, driver):self.driver = driverself.driver.get("https://example.com")@handle_element_exceptions(retry_times=2, default_value="无搜索结果")def get_search_result_text(self):"""获取搜索结果文本,可能返回空字符串或抛出异常"""element = self.driver.find_element(By.CSS_SELECTOR, "#search-result")return element.text.strip() # 可能返回空字符串(业务空值)
测试用例层逻辑:
def test_search_functionality():driver = webdriver.Chrome()search_page = SearchPage(driver)result = search_page.get_search_result_text()assert result != "无搜索结果", "搜索功能异常" # 直接断言有效值driver.quit()
三、避免测试用例冗余的核心原则
1. 三层架构设计
层级 | 职责描述 | 异常处理位置 |
---|---|---|
测试用例层 | 业务流程验证、断言 | 无try/catch |
中间层(Page Object) | 具体UI操作(元素定位、点击) | 装饰器应用层 |
框架底层 | 通用工具(驱动管理、日志、截图) | 装饰器定义层 |
2. 装饰器的优势对比
传统try/catch | 装饰器方案 |
---|---|
测试用例代码臃肿 | 代码简洁,聚焦业务逻辑 |
异常处理逻辑分散 | 集中管理,修改成本低 |
难以复用 | 可复用至所有类似操作 |
四、多项目场景下的装饰器实践:策略模式与动态配置
当多个项目共用自动化框架时,通过策略模式实现差异化异常处理:
1. 定义项目级异常处理策略
# project_strategies.py
STRATEGY_A = { # 电商项目:严格重试+默认空值"retry_times": 3,"default_value": "商品无库存"
}STRATEGY_B = { # 后台项目:快速失败+截图"retry_times": 0,"default_value": "数据为空","screenshot_path": "/error_logs/"
}
2. 装饰器动态加载策略
def load_strategy(project_name):strategies = {"project_a": STRATEGY_A,"project_b": STRATEGY_B}return strategies.get(project_name, {})def ui_action_with_strategy(project_name):strategy = load_strategy(project_name)return handle_element_exceptions(**strategy) # 解包策略参数
3. 项目级Page Object应用
# project_a/pages.py
class ProductPage:@ui_action_with_strategy("project_a")def check_stock(self):return self.driver.find_element(By.CSS_SELECTOR, ".stock").text# project_b/pages.py
class ReportPage:@ui_action_with_strategy("project_b")def get_report_data(self):return self.driver.find_element(By.ID, "data").text
五、总结:装饰器的核心价值
- 解耦业务逻辑与技术细节:测试用例仅关注“验证什么”,框架层处理“如何处理异常”
- 提升可维护性:异常处理逻辑集中在装饰器中,修改时无需触及测试用例
- 支持复杂场景:通过参数化和策略模式,轻松应对多项目、多场景需求
最佳实践建议:
- 装饰器命名遵循
handle_xxx
或xxx_strategy
规范 - 用
functools.wraps
保留原函数元信息 - 在框架底层统一管理装饰器,避免重复定义
通过装饰器模式,UI自动化测试代码将更简洁、更健壮,真正实现“一次编写,多处复用”的工程化目标。
相关文章:
从装饰器出发,优雅处理 UI 自动化中的异常
二、装饰器核心原理:函数式编程的“语法糖” 在 UI 自动化测试的实际场景中,页面加载超时、元素定位失败、网络波动等异常问题频繁出现。传统的try-except嵌套方式虽然能捕获异常,但会导致业务代码与处理逻辑高度耦合,造成代码冗…...
华为首款鸿蒙电脑正式亮相
在2025年5月8日的鸿蒙电脑技术与生态沟通会上,华为终端BG平板与PC产品线总裁朱懂东发表演讲。鸿蒙办公将支持无缝协同(统一账户 / 多端同步)、智慧体验(鸿蒙 AI / 智慧交互)、融合生态(桌面生态 / 移动生态…...
森林生态学研究深度解析:R语言入门、生物多样性分析、机器学习建模与群落稳定性评估
在生态学研究中,森林生态系统的结构、功能与稳定性是核心研究内容之一。这些方面不仅关系到森林动态变化和物种多样性,还直接影响森林提供的生态服务功能及其应对环境变化的能力。森林生态系统的结构主要包括物种组成、树种多样性、树木的空间分布与密度…...
需求分析阶段测试工程师主要做哪些事情
在软件测试需求分析阶段,主要围绕确定测试范围、明确测试目标、细化测试内容等方面开展工作,为后续测试计划的制定、测试用例的设计以及测试执行提供清晰、准确的依据。以下是该阶段具体要做的事情: 1. 需求收集与整理 收集需求文档&#x…...
IBM BAW(原BPM升级版)使用教程第五讲
结前篇! 一、服务:外部服务 在 IBM Business Automation Workflow (BAW) 中,外部服务(External Services)是指在流程中调用和集成外部系统或服务的组件。外部服务允许IBM BAW与其他业务系统、应用程序或第三方服务进行…...
一个用C#开发的记事本Notepads开源编辑器
从零学习构建一个完整的系统 推荐一个用C#开发的,模仿Nodepad的记事本编辑器。 项目简介 这是一个国人开发的,采用Fluent Design风格,支持Markdown语法及其预览,可以用于替代Windows自带记事本功能。 虽然它没有Nodepad、VS C…...
Office宏病毒钓鱼-打点突破
Office宏钓鱼 [攻击路径] / [利用流程] - 详细步骤:制作一个简单的Word宏钓鱼文档 步骤一:规划恶意宏功能 (Payload Design) 目的: 决定宏执行后要达成的目标。为教学演示,我们设定一个简单的目标:弹出一个消息框&…...
神经网络—感知器、多层感知器
文章目录 前言一、生物神经元与感知器的类比二、感知器1、简单感知器2、多层感知器(1)多层感知机结构 3、神经网络结构 总结1、感知器的局限性如何突破感知器的局限性? 2、感知器的应用 前言 感知器(Perceptron)是神经…...
美化IDEA注释:Idea 中快捷键 Ctrl + / 自动注释的缩进(避免添加注释自动到行首)以及 Ctrl + Alt + l 全局格式化代码的注释缩进
打开 Settings 界面,依次选择 Editor -> Code Style -> Java,选择 Code Generation, 取消 Line comment at first column 和 Block comment at first column 的勾选即可, 1、Line comment at first column (行注释在第一列…...
基于大型语言模型的高效时间序列预测模型选择
时间序列预测在金融、医疗、能源等多个领域具有重要意义,其结果对于决策和规划起到关键作用。然而,选择合适的预测模型往往需要丰富的领域知识和大量计算资源。Abdallah等(2022)的研究表明,没有单一算法能在所有预测任…...
2025 EAU UTUC指南学习笔记③:诊断策略精读——从症状到活检,如何科学判断治疗路径?
👋 欢迎关注我的学习专栏~ 如果觉得文章有帮助,别忘了点赞、关注、评论,一起学习 本文聚焦UTUC的诊断策略,全面梳理影像学检查、尿液学检测、内镜评估与活检手段,并结合指南推荐等级,提出科研视角的思考。 …...
【Ubuntu】Netplan静态网络配置
1.配置文件 # 默认配置文件:/etc/netplan/*.yaml# 本机 rootubuntu-server:~# vim /etc/netplan/50-cloud-init.yaml 2.DHCP network: ethernets:ens33:dhcp4: trueversion: 2 3.静态IP设置 注意:netplan 说明文件格式存储在下列路径下…...
【SQLSERVER】Ubuntu 连接远程 SQL Server(MSSQL)
Ubuntu 连接远程 SQL Server(MSSQL) 引言 在企业级开发环境中,跨平台的数据库连接是常见需求。虽然 SQL Server 通常在 Windows 服务器上运行,但在 Ubuntu 上连接远程 SQL Server 也非常重要。本文将详细介绍如何在 Ubuntu 上通…...
GuPPy-v1.2.0安装与使用-生信工具52
GuPPy:Python中用于光纤光度数据分析的免费开源工具 01 背景 Basecalling 是将原始测序信号转换为碱基序列的过程,通俗地说,就是“把碱基识别出来”。这一过程在不同代测序技术中各不相同: 一代测序是通过解析峰图实现࿱…...
Mysql-OCP PPT课程讲解并翻译
#跳过介绍,直接从干货开始记录 第一章 安装mysql...
maven 依赖冲突异常分析
异常如下 Description:An attempt was made to call a method that does not exist. The attempt was made from the following location:com.google.common.cache.CacheBuilder.initialCapacity(CacheBuilder.java:353)The following method did not exist:com.google.common…...
【软件设计师:软件工程】9.软件开发模型与方法
一、软件危机与软件工程 软件危机与软件工程是计算机科学发展中密切相关的两个概念。 1.软件危机(Software Crisis) 背景:20世纪60年代至80年代,随着计算机硬件性能提升,软件规模与复杂度剧增,传统开发方法难以应对,导致大量项目失败。 表现: 成本失控:开发周期长、…...
用uniapp在微信小程序实现画板(电子签名)功能,使用canvas实现功能
效果: 功能:实现重签 退出 保存 等功能 解决的问题: 电子签名画布抖动问题解 注意: 保存的时候上传到自己的服务器地址,后端返回图片地址 代码: <template><view><view class"signature&qu…...
旅游设备生产企业的痛点 质检系统在旅游设备生产企业的应用
在旅游设备制造行业,产品质量直接关系到用户体验与企业口碑。从景区缆车、观光车到水上娱乐设施,每一件设备的安全性与可靠性都需经过严苛检测。然而,传统质检模式常面临数据分散、流程不透明、合规风险高等痛点,难以满足旅游设备…...
为什么使用Less替代原始CSS?
What is Less? Less 是一种 CSS 预处理器,它在 CSS 的基础上添加了许多有用的特性,提供了更高效、更灵活的方式来编写样式代码。 特性: 1、变量(Variables) 优势: 可以定义一个值并在整个样…...
C++ - 类和对象 #日期类的实现
文章目录 前言 一、导言 二、构造 三、比较大小 1、实现大于 2、等于 3、大于等于 4、小于 5、小于等于 6、不等于 二、加减 1、加与加等 2、减与减等 3、、-- 4、日期-日期 三、流提取、流插入 1、流插入 2、流提取 四、日期类所有代码汇总 总结 前言 路…...
6:点云处理—QT三维窗口显示
1.编写halcon显示程序 导出C格式 *读取三维点云 dev_get_window(WindowHandle)dev_open_window(0, 0, 512, 512, black, WindowHandle1)*512, 512 CamParam : [0.01,0,7e-6,7e-6,512/2, 512/2,512, 512] DisPose:[30.427, -1.05274, 97.8798, 8.29326, 356.416, 99.016, 0]*彩…...
css3伸缩盒模型第三章(伸缩相关)
伸缩盒模型第三章(伸缩) 介绍了伸缩性的伸缩提现到哪些方面,以及flex 居中的小技巧,视觉差异等 flex 水平垂直居中伸缩性flex 复合式属性项目排序侧轴单独对齐 flex 水平居中 我们可以使用flex 使元素居中对齐,当然也可以用 margin 等进行…...
【Ubuntu】安裝向日葵远程控制
前言 在Ubuntu 24.04.2下安装向日葵远程控制出错,少了一些依赖,需要安装一些依赖。 1.安装gconf2-common wget http://mirrors.kernel.org/ubuntu/pool/universe/g/gconf/gconf2-common_3.2.6-6ubuntu1_all.deb sudo dpkg -i gconf2-common_3.2.6-6ub…...
Screeps Arena基础入门
本文主要内容 JavaSsript语法使用VScode编译环境Screeps Arena游戏规则 JavaSsript语法使用 基本数据类型 // String, Numker,Boolean,null, undefined const username "John"; const age 30; const rate 4.5; const iscool true; const x null; #表示值为…...
docker 安装 sqlserver2022 和注意点
一、前言 1、可以直接参考微软官方文档 快速入门:使用 Docker 运行 SQL Server Linux 容器映像,这里主要是说一些注意点和坑 二、安装 1、拉取镜像 docker pull mcr.microsoft.com/mssql/server:2022-latest2、创建挂载目录,这里只是比官方…...
Centos系统详解架构详解
CentOS 全面详解 一、CentOS 概述 CentOS(Community Enterprise Operating System) 是基于 Red Hat Enterprise Linux(RHEL) 源代码构建的免费开源操作系统,专注于稳定性、安全性和长期支持,广泛应用于服…...
Git实战经验分享:深入掌握git commit --amend的进阶技巧
一、工具简介 git commit --amend是Git版本控制系统的核心补救命令,主要用于修正最近一次提交的元数据。该命令不会产生新的提交记录,而是通过覆盖原提交实现版本历史的整洁性,特别适合在本地仓库进行提交优化。 二、核心应用场景 提交信息…...
基于LSTM与SHAP可解释性分析的神经网络回归预测模型【MATLAB】
基于LSTM与SHAP可解释性分析的神经网络回归预测模型【MATLAB】 一、引言 在数据驱动的智能时代,时间序列预测已成为许多领域(如金融、气象、工业监测等)中的关键任务。长短期记忆网络(LSTM)因其在捕捉时间序列长期依…...
C++卡特兰数讲解
前情提要,参考资料:卡特兰数 - OI Wiki 一、定义 卡特兰数(Catalan number)是一个在组合数学中经常出现的数列,应用范围很广,例如括号匹配问题、出栈顺序问题、多边形三角剖分问题等。在 C 中,可以使用多种…...
【数据融合实战手册·应用篇】“数字孪生+视频融合”让智慧城市拥有空间感知
一、视频融合技术如何破局城市治理? #从"碎片监控"到"上帝视角" 传统视频监控系统画面分散,监管人员需要观看多个分镜头画面,难以将零散的分镜头视频与其实际地理位置对应,容易产生信息孤岛,同时…...
[数据库之十一] 数据库索引之联合索引
执行数据库查询时,通常查询条件是多对个属性进行判断和约束,对于这种类型的查询,如果存在多个索引则使用多个索引,或者使用建立在多属性搜索码上的索引,这样能提高查询效率。 一、使用多个单码索引 假设数据表 instruc…...
【Linux】Linux工具(1)
3.Linux工具(1) 文章目录 3.Linux工具(1)Linux 软件包管理器 yum什么是软件包关于 rzsz查看软件包——yum list命令如何安装软件如何卸载软件补充——yum如何找到要安装软件的下载地址 Linux开发工具Linux编辑器-vim使用1.vim的基…...
微信小程序使用腾讯云COS SDK实现用户头像上传
引入与初始化COS SDK (从官网下载cos-wx-sdk-v5.js文件,本地上传) const COS require(../../../static/utils/cos-wx-sdk-v5.js) as any;//实例化一个cos对象 const cos new COS({SecretId: AK***********************VedSJ, //idSecretK…...
什么是HTML、CSS 和 JavaScript?
HTML、CSS 和 JavaScript 是构建网页的三大核心技术,它们分工明确又紧密协作。接下来我将分别介绍三者的定义、功能,并阐述它们如何共同构成网页,最后推荐学习资源。 一、HTML:网页的骨架与内容基础 HTML(HyperText …...
兼具直连存储与分布式存储优势的混合存储方案
兼具直连存储与分布式存储优势的混合存储方案 一、分层存储架构 热数据本地直存 将高频访问的热数据(如数据库事务日志、实时计算中间结果)存储在DAS本地SSD中利用DAS的低延迟特性提升实时性要求高的业务性能 温冷数据分布式存储 低频访问的温…...
App Store支付新政重构跨境电商生态:eBay卖家的突围之道
2024年4月,美国联邦法院对苹果应用商店支付体系的裁决掀起全球电商海啸。这场源自Epic Games诉讼案的制度变革,正在跨境电商领域掀起超预期的蝴蝶效应。本文将深入剖析政策变革对eBay生态的冲击路径,揭示跨境卖家转型升级的核心策略。 一、支…...
Rust 开发环境搭建与插件
生成完整文章 用户请求写一篇完整的CSDN文章,要求根据之前的建议生成。文章要包含结构清晰、标题、代码示例、常见提示、跨平台Rust工具链安装差异等内容。文章需准确,包括插件和使用提示,CLI程序示例等。主要内容包括:介绍、Rus…...
Axios替代品Alova
介绍alova | Alova.JS Multipart 实体请求 | Axios中文文档 | Axios中文网 1. 极致的轻量与性能 Tree-shaking优化:仅打包使用到的功能模块 零依赖:基础包仅 4KB(Axios 12KB) 2. 智能请求管理(开箱即用࿰…...
【C语言】文件操作(续)
目录 复习: 一⽂件的顺序读写 例子: 前言: 在上篇文章中介绍了文件的类型,文件指针,流,操作的函数。 在本篇文章继续为大家带来文件细节分享,如 顺序读写等等。 复习: fopen是…...
Angular 面试常见问题
1. 请阐述 Angular 的工作原理 Angular 的工作流程涉及多个关键环节,从组件交互到浏览器渲染,以下是其核心流程: 组件交互:当用户触发特定事件(如点击按钮)时,组件会响应这些交互,…...
数据库(MySQL)基础
一、登录数据库 在linux系统中登录数据库的指令 mysql -h 127.48.0.236 -P 3306 -u root -p -h:填写IP地址,指明要连接的主机。如果不加该字段表示本地主机-P:填写端口号,指明进程。 如果不加该字段会使用默认的端口号。-u&…...
【Java ee 初阶】文件操作和IO(上)
一、文件 文件在计算机中,是保存到“硬盘”上的。操作系统,把硬盘操作进行了抽象封装,使得编程的时候,是不会直接操作硬盘的,而是通过“文件”的概念来进行间接操作。 文件有哪些操作?——>打开文件&a…...
微信小程序备案的一些记录
小程序如果没有备案是搜索不到小程序的。 小程序备案需要填写主体负责人的信息,需要主体负责人的手机号验证码, 需要填写管理员的信息,同样也需要验证手机号码, 填写完毕之后,提交进行初审,初审之后会打…...
leetcode0279. 完全平方数-medium
1 题目:完全平方数 官方标定难度:中 给你一个整数 n ,返回 和为 n 的完全平方数的最少数量 。 完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,1…...
2018机械行业ERP软件发展趋势
随着互联网经济的发展,实体的经济将来很有发展的优势,管理的信息化工具,也要随着市场需求的改变而改变。 以前的ERP管理系统,管理管控的方向。 1、以物料管理为核心,通过ERP管理系统,将企业的物料管理清楚&…...
限制布局大小,实现文本自适应
实现数字部分自适应 适配后 使用页需绑定ref <div class"setting-bind-text" ref"element" :style"{ transform: scale(${scale}) }">{{ coin }}</div> script部分引入使用 import { useTextScale } from /hooks/useTextScale; c…...
涨薪技术|0到1学会性能测试第52课-Tomcat调优技术
前面的推文我们掌握了Tomcat服务器的3种监控技术知识。今天给大家分享Tomcat调优技术。后续文章都会系统分享干货,带大家从0到1学会性能测试。 在对Tomcat进行调优之前,需要对Tomcat的结构体系有一个清楚的了解,这对调优起到至交重要的作用,Tomcat结构体系图,如图10-20所示…...
Arm核的Ubuntu系统上安装Wireshark
Arm核的Ubuntu系统上安装Wireshark 一、安装wireshark 安装命令: sudo apt-get install wireshark-qt 如下图所示: 安装过程弹出如下界面: 鼠标选择Yes,点回车键确认 安装完成。 二、打开wireshark 输入命令行打开wireshark …...
C++模板【上篇】 —详解模板基础语法
文章目录 前言1. 泛型编程2. 模板的类别2.1 函数模板2.2 类模板 3. 模板的实例化3.1 函数模板的实例化3.1.1 隐式实例化* 编译器实例化原理3.1.2 显示实例化 3.2 类模板的实例化 前言 在这篇文章中,主要介绍一些模板的基础的语法和一些细节,同时了解泛型…...