当前位置: 首页 > news >正文

Selenium 二次封装通用页面基类 BasePage —— Python 实践

🧾  一、项目背景

        在自动化测试中,页面对象模型(Page Object Model) 是一种非常重要的设计模式,它将页面元素和操作封装成类,提升代码复用性、可维护性和可读性。
        本文将以一个完整的 BasePage 页面基类 实现为例,详细讲解如何构建一个结构清晰、功能强大的 Selenium 页面基类,并结合日志记录、截图、等待等常用功能进行二次封装,为后续编写测试用例打下坚实基础。

🧱 二、项目结构概览

Auto_selenium/
├── page_objects/
│   └── base_page.py          # 页面基类
├── utils/
│   └── logger.py             # 日志工具类
├── test_cases/               # 测试用例目录
└── run_all_tests.py          # 测试执行入口

我们将在 base_page.py 中实现一个通用的 BasePage 类,供所有页面继承使用。

🛠️ 三、BasePage 类详解

1. 初始化方法

def __init__(self, driver):self.driver = driver
  • 构造函数接收浏览器驱动作为参数,每个页面对象都将拥有对浏览器的访问能力。
  • 这是 Page Object 模式的基础。

2. 打开网页

def open(self, url):self.driver.get(url)
  • 封装 get() 方法,用于打开指定 URL。
  • 示例调用:
page.open("https://www.baidu.com")

3. 浏览器前进与后退

def forward(self):self.driver.forward()logger.info("在当前页面上单击前进......")def back(self):self.driver.back()logger.info("点击返回当前页面......")
  • 控制浏览器前进、后退操作。
  • 结合日志输出,便于调试和报告生成。

4. 显式等待

def wait(self, loc, seconds):try:wait_ = WebDriverWait(self.driver, seconds)wait_.until(lambda driver: driver.find_element(*loc))logger.info("等待 %d 秒......" % seconds)except NameError as e:logger.error("未能加载元素 %s" % e)
  • 使用 WebDriverWait 等待元素出现,避免因元素未加载完成导致查找失败。
  • 推荐替代 time.sleep(),提高脚本健壮性。

5. 截图功能

    def get_browser_img(self, timestamp=None):"""截图并保存到项目根目录下的 screenshots 文件夹中。:param timestamp: 可选参数,用于指定截图文件名的时间戳"""try:base_dir = os.path.dirname(os.path.abspath('__file__'))  # 项目根目录screenshot_dir = os.path.join(base_dir, 'screenshots')# 自动创建目录(如果不存在)os.makedirs(screenshot_dir, exist_ok=True)# 如果未提供 timestamp,则使用当前时间if timestamp is None:timestamp = time.strftime('%Y%m%d%H%M%S', time.localtime(time.time()))# 构建文件路径screen_name = os.path.join(screenshot_dir, f"{timestamp}.png")# 执行截图并保存self.driver.get_screenshot_as_file(screen_name)logger.info("已截图并保存到文件夹 : /screenshots")except Exception as e:logger.error("截图失败! %s" % e)self.get_browser_img(timestamp)     # 递归调用,重新截图
  • 自动创建 screenshots 文件夹。
  • 支持自定义时间戳命名文件。
  • 出错时自动截图并递归重试。

✅ 建议:可在测试失败时自动调用该方法,辅助定位问题。

6. 元素定位与操作

定位元素

def find_element(self, loc):return self.driver.find_element(*loc)
  • 返回一个 WebElement 对象,支持多种定位方式 (By.ID, 'id')。

输入文本

def send_keys(self, selector, text):el = self.find_element(selector)el.clear()try:el.send_keys(text)logger.info("在输入框中输入 \' %s \' " % text)except NameError as e:logger.error("在输入框中选择 %s 失败" % e)self.get_browser_img()
  • 清空输入框后输入内容。
  • 异常处理+截图机制,确保稳定性。

清除文本

def clear(self, selector):el = self.find_element(selector)try:el.clear()logger.info("输入前清除输入框中的文本......")except NameError as e:logger.error("Failed to clear in input box with %s" % e)self.get_browser_img()
  • 防止旧数据干扰新操作。

点击元素

def click(self, selector):el = self.find_element(selector)try:el.click()logger.info("点击元素 \'%s\' " % el.text)except NameError as e:logger.error("Failed to click the element with %s" % e)self.get_browser_img()
  • 可记录点击的文本信息,方便日志分析。

7. 鼠标操作(悬停+点击)

def move_element(self, loc, sloc):mouse = self.find_element(loc)try:ActionChains(self.driver).move_to_element(mouse).perform()self.click(sloc)except Exception as e:logger.error("Failed to click move_element with %s" % e)self.get_browser_img()
  • 适用于菜单、下拉框等需要鼠标悬停才能点击的场景。
  • 支持链式操作 + 错误处理。

8. 强制等待(静态方法)

@staticmethod
def sleep(seconds):time.sleep(seconds)logger.info("强制等 %d 秒" % seconds)
  • 使用 @staticmethod 装饰器,不依赖实例或类。
  • 适合全局使用的工具方法。

📦 四、如何在实际测试中使用?

示例:百度搜索页面类

from page_objects.base_page import BasePageclass BaiduSearchPage(BasePage):def search(self, keyword):self.send_keys(("id", "kw"), keyword)self.click(("id", "su"))

示例:测试用例调用

from selenium import webdriver
from page_objects.baidu_search_page import BaiduSearchPagedriver = webdriver.Chrome()
page = BaiduSearchPage(driver)page.open("https://www.baidu.com")
page.search("Selenium")

📝 五、总结

        该 BasePage 页面基类为 Selenium 自动化测试提供了良好的封装基础,包含了浏览器操作、元素定位、等待机制、截图功能及日志记录等常用操作。通过继承该基类,可提升测试代码的复用性与可维护性,同时结合异常处理和工具方法,增强了脚本的稳定性与可读性,是构建高效自动化测试框架的重要组成部分。

http://www.lryc.cn/news/575251.html

相关文章:

  • GBDT:梯度提升决策树——集成学习中的预测利器
  • Git上传代码如何解决Merge冲突
  • 时序数据库 TDengine 助力华锐 D5 平台实现“三连降”:查询快了,机器少了,成本也低了
  • 【目标检测】平均精度(AP)与均值平均精度(mAP)计算详解
  • MicroPython网络编程:AP模式与STA模式详解
  • 大塘至浦北高速分布式光伏项目,让‘交通走廊’变身‘绿色能源带’
  • 深度学习入门--(二)感知机
  • python的kivy框架界面布局方法详解
  • react中使用3D折线图跟3D曲面图
  • Vue Devtools “Open in Editor” 配置教程(适用于 VSCode 等主流编辑器)
  • 大语言模型(LLM)初探:核心概念与应用场景
  • 【MongoDB】MongoDB从零开始详细教程 核心概念与原理 环境搭建 基础操作
  • DeepSeek模型接入LangChain流程(详细教程)
  • 永磁同步电机无速度算法--基于同步旋转坐标系锁相环的滑模观测器
  • PYTHON从入门到实践6-字典
  • MCP2518FD发送时有时候多发数据包问题
  • 【预告 大模型应用开发实战专栏 升级】将增加《大模型 Agent 应用实战指南》专题赋能 Agent 开发者
  • OpenGL模板缓冲:实现亮显外轮廓效果
  • C# LINQ语法
  • Python 爬虫入门:从数据爬取到转存 MySQL 数据库
  • Cookie 在 HTTP 中的作用HTTP 中的状态码
  • 北斗导航 | 基于改进奇偶矢量法的CAT I精密进近RAIM算法
  • 半导体芯闻--20250625
  • Linux离线安装jdk-11
  • AudioTrack使用
  • Kylin Linux Advanced Server V10 离线安装 Prometheus + Grafana + node_exporter指南
  • 【网站内容安全检测】之1:获取网站所有链接sitemap数据
  • Sortablejs动态同类型穿插
  • MySQL之视图深度解析
  • 灰度发布怎么保证数据库一致的