从零开始学习:如何使用Selenium和Python进行自动化测试?
安装selenium
打开命令控制符输入:pip install -U selenium
火狐浏览器安装firebug:www.firebug.com,调试所有网站语言,调试功能
Selenium IDE 是嵌入到Firefox 浏览器中的一个插件,实现简单的浏览器操 作的录制与回放功能,IDE 录制的脚本可以可以转换成多种语言,从而帮助我们快速的开发脚本,下载地址:https://addons.mozilla.org/en-US/firefox/addon/selenium-ide/
如何使用IDE录制脚本:点击seleniumIDE——点击录制——开始录制——录制完成后点击文件Export Test Case——python/unittest/Webdriver——保存;
安装python
安装的时候,推荐选择“Add exe to path”,将会自动添加Python的程序到环境变量中。然后可以在命令行输入 python -V 检测安装的Python版本。
浏览器内壳:IE、chrome、FireFox、Safari
1、webdriver:用unittest框架写自动化用例(setUp:前置条件,tearDown清场)
1 import unittest2 from selenium import webdriver3 4 class Ranzhi(unittest.TestCase):5 def setUp(self):6 self.driver = webdriver.Firefox() #选择火狐浏览器7 def test_ranzhi(self):8 pass9 def tearDown(self):
10 self.driver.quit()#退出浏览器
2、断言,检查跳转的网页是否和实际一致
断言网址时需注意是否为伪静态(PATH_INFO)或者GET,前者采用路径传参数(sys/user-creat.html),后者通过字符查询传参数(sys/index.php?m=user&f=index)
当采用不同方式校验网址会发现变化。
1 self.assertEqual("http://localhost:8080/ranzhi/www/s/index.php?m=index&f=index",
2 self.driver.current_url, "登录跳转失败")
3、定位元素,在html里面,元素具有各种各样的属性。我们可以通过这样唯一区别其他元素的属性来定位到这个元素.
WebDriver提供了一系列的元素定位方法。常见的有以下几种:id,name,link text,partial link text,xpath,css seletor,class,tag.
1 self.driver.find_element_by_xpath('//*[@id="s-menu-superadmin"]/button').click()
2 self.driver.find_element_by_id('account').send_keys('admin')
3 self.driver.find_element_by_link_text(u'退出').click()
定位元素需注意的问题:
a.时间不够,采用两种方式(self.implicitly_wait(30),sleep(2))
b.函数嵌套(<iframe></iframe>)
1 # 进入嵌套
2 self.driver.switch_to.frame('iframe-superadmin')
3 #退出嵌套
4 self.driver.switch_to.default_content()
c.flash,验证码(关闭验证码或使用万能码)
d.xpath问题:最好采用最简xpath,当xpath中出现li[10]等时需注意,有时页面定位会出现问题
4、采用CSV存数据
CSV:以纯文本形式存储表格数据(数字和文本),CSV文件由任意数目的记录组成,记录间以某种换行符分隔;每条记录由字段组成,字段间的分隔符是其它字符或字符串,最常见的是逗号或制表符。大量程序都支持某种CSV变体,至少是作为一种可选择的输入/输出格式。
1 melody101,melody101,m,1,3,123456,@qq.com
2 melody102,melody101,f,2,5,123456,@qq.com
3 melody103,melody101,m,3,2,123456,@qq.com
1 import csv2 # 读取CSV文件到user_list字典类型变量中3 user_list = csv.reader(open("list_to_user.csv", "r"))4 # 遍历整个user_list5 for user in user_list:6 sleep(2)7 self.logn_in('admin', 'admin')8 sleep(2)9 # 读取一行csv,并分别赋值到user_to_add 中
10 user_to_add = {'account': user[0],
11 'realname': user[1],
12 'gender': user[2],
13 'dept': user[3],
14 'role': user[4],
15 'password': user[5],
16 'email': user[0] + user[6]}
17 self.add_user(user_to_add)
5、对下拉列表的定位采用select标签
1 from selenium.webdriver.support.select import Select
2 # 选择部门
3 dp =self.driver.find_element_by_id('dept')
4 Select(dp).select_by_index(user['dept'])
5 # 选择角色
6 Select(self.driver.find_element_by_id('role')).select_by_index(user['role'])
6、模块化代码
需要对自动化重复编写的脚本进行重构(refactor),将重复的脚本抽取出来,放到指定的代码文件中,作为共用的功能模块。使用模块化代码注意需倒入该代码。
1 #模块化代码后引用,需导入代码模块2 from ranzhi_lib import RanzhiLib3 self.lib = RanzhiLib(self.driver)4 # 点击后台管理5 self.lib.click_admin_app()6 sleep(2)7 # 点击添加用户8 self.lib.click_add_user()9 # 添加用户
10 self.lib.add_user(user_to_add)
11 sleep(1)
12 # 退出
13 self.lib.logn_out()
14 sleep(2)
1 class RanzhiLib():
2 # 构造方法
3 def __init__(self, driver):
4 self.driver = driver
7、自定义函数运行的先后顺序:完整的单元测试很少只执行一个测试用例,开发人员通常都需要编写多个测试用例才能对某一软件功能进行比较完整的测试,这些相关的测试用例称为一个测试用例集,在PyUnit中是用TestSuite类来表示,采用unittest.TestSuite()。
PyUnit使用TestRunner类作为测试用例的基本执行环境,来驱动整个单元测试过程。Python开发人员在进行单元测试时一般不直接使用TestRunner类,而是使用其子类TextTestRunner来完成测试。
详情请查看:http://www.ibm.com/developerworks/cn/linux/l-pyunit/
1 # 构造测试集
2 suite = unittest.TestSuite()
3 suite.addTest(RanzhiTest("test_login"))
4 suite.addTest(RanzhiTest("test_ranzhi"))
5
6 # 执行测试
7 runner = unittest.TextTestRunner()
8 runner.run(suite)
以下代码为登录“然之系统”,进入添加用户,循环添加用户并检测添加成功,再退出的过程。以下程序分别为主程序,模块化程序,执行程序,CSV文件
1 import csv2 import unittest3 from time import sleep4 5 from selenium import webdriver6 # 模块化代码后引用需导入代码模块7 from ranzhi_lib import RanzhiLib8 9
10 class Ranzhi(unittest.TestCase):
11 def setUp(self):
12 self.driver = webdriver.Firefox()
13 self.lib = RanzhiLib(self.driver)
14
15 # 主函数
16 def test_ranzhi(self):
17 # 读取CSV文件到user_list字典类型变量中
18 user_list = csv.reader(open("list_to_user.csv", "r"))
19 # 遍历整个user_list
20 for user in user_list:
21 sleep(2)
22 self.lib.logn_in('admin', 'admin')
23 sleep(2)
24 # 断言
25 self.assertEqual("http://localhost:8080/ranzhi/www/sys/index.html",
26 self.driver.current_url,
27 '登录跳转失败')
28 # 读取一行csv,并分别赋值到user_to_add 中
29 user_to_add = {'account': user[0],
30 'realname': user[1],
31 'gender': user[2],
32 'dept': user[3],
33 'role': user[4],
34 'password': user[5],
35 'email': user[0] + user[6]}
36 # 点击后台管理
37 self.lib.click_admin_app()
38 # 进入嵌套
39 self.lib.driver.switch_to.frame('iframe-superadmin')
40 sleep(2)
41 # 点击添加用户
42 self.lib.click_add_user()
43 # 添加用户
44 self.lib.add_user(user_to_add)
45 # 退出嵌套
46 self.driver.switch_to.default_content()
47 sleep(1)
48 # 退出
49 self.lib.logn_out()
50 sleep(2)
51 # 用新账号登录
52 self.lib.logn_in(user_to_add['account'], user_to_add['password'])
53 sleep(2)
54 self.lib.logn_out()
55 sleep(2)
56
57 def tearDown(self):
58 self.driver.quit()
1 from time import sleep2 3 from selenium.webdriver.support.select import Select4 5 6 class RanzhiLib():7 # 构造方法8 def __init__(self, driver):9 self.driver = driver
10
11 # 模块化添加用户
12 def add_user(self, user):
13 driver = self.driver
14 # 添加用户名
15 ac = driver.find_element_by_id('account')
16 ac.send_keys(user['account'])
17 # 真实姓名
18 rn = driver.find_element_by_id('realname')
19 rn.clear()
20 rn.send_keys(user['realname'])
21 # 选择性别
22 if user['gender'] == 'm':
23 driver.find_element_by_id('gender2').click()
24 elif user['gender'] == 'f':
25 driver.find_element_by_id('gender1').click()
26 # 选择部门
27 dp = driver.find_element_by_id('dept')
28 Select(dp).select_by_index(user['dept'])
29 # 选择角色
30 role = driver.find_element_by_id('role')
31 Select(role).select_by_index(user['role'])
32 # 输入密码
33 pwd1 = driver.find_element_by_id('password1')
34 pwd1.clear()
35 pwd1.send_keys(user['password'])
36
37 pwd2 = driver.find_element_by_id('password2')
38 pwd2.send_keys(user['password'])
39 # 输入邮箱
40 em = driver.find_element_by_id('email')
41 em.send_keys(user['email'])
42 # 点击保存
43 driver.find_element_by_id('submit').click()
44 sleep(2)
45
46 # 登录账号
47 def logn_in(self, name, password):
48 driver = self.driver
49 driver.get('http://localhost:8080/ranzhi/www')
50 sleep(2)
51
52 driver.find_element_by_id('account').clear()
53 driver.find_element_by_id('account').send_keys(name)
54 driver.find_element_by_id('password').clear()
55 driver.find_element_by_id('password').send_keys(password)
56 driver.find_element_by_id('submit').click()
57 sleep(2)
58
59 # 退出账号
60 def logn_out(self):
61 self.driver.find_element_by_id('start').click()
62 sleep(4)
63 self.driver.find_element_by_link_text(u'退出').click()
64 sleep(3)
65
66 # 点击后台管理
67 def click_admin_app(self):
68 self.driver.find_element_by_xpath('//*[@id="s-menu-superadmin"]/button').click()
69 sleep(1)
70
71 def click_add_user(self):
72 self.driver.find_element_by_xpath('//*[@id="shortcutBox"]/div/div[1]/div/a/h3').click()
73 sleep(3)
1 import unittest2 3 from ranzhi import Ranzhi4 5 6 class RanzhiTestRunner():7 def run_tests(self):8 suite = unittest.TestSuite()9 suite.addTest(Ranzhi('test_ranzhi'))
10 runner = unittest.TextTestRunner()
11 runner.run(suite)
12
13
14 if __name__ == "__main__":
15 ranzhi_test_runner = RanzhiTestRunner()
16 ranzhi_test_runner.run_tests()
1 melody109,melody101,m,1,3,123456,@qq.com
2 melody106,melody101,f,2,5,123456,@qq.com
3 melody107,melody101,m,3,2,123456,@qq.com
行动吧,在路上总比一直观望的要好,未来的你肯定会感 谢现在拼搏的自己!如果想学习提升找不到资料,没人答疑解惑时,请及时加入扣群: 320231853,里面有各种软件测试+开发资料和技术可以一起交流学习哦。
最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:
这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!
相关文章:

从零开始学习:如何使用Selenium和Python进行自动化测试?
安装selenium 打开命令控制符输入:pip install -U selenium 火狐浏览器安装firebug:www.firebug.com,调试所有网站语言,调试功能 Selenium IDE 是嵌入到Firefox 浏览器中的一个插件,实现简单的浏览器操 作的录制与回…...

【linux基础】bash脚本的学习:定义变量及引用变量、统计目标目录下所有文件行数、列数
假设目的:统计并输出指定文件夹下所有文件行数 单个文件可以用 wc -l ;多个文件,可以用通配符 / 借助bash脚本 1.定义变量名,使用引号 a"bestqc.com.map" b"Anno.variant_function" c"enrichment/GOe…...

算法四十天-删除排序链表中的重复元素
删除排序链表中的重复元素 题目要求 解题思路 一次遍历 由于给定的链表是排好序的,因此重复的元素在链表中的出现的位置是连续的,因此我们只需要对链表进行一次遍历,就可以删除重复的元素。 具体地,我们从指针cur指向链表的头节…...

Linux-等待子进程
参考资料:《Linux环境编程:从应用到内核》 僵尸进程 进程退出时会进行内核清理,基本就是释放进程所有的资源,这些资源包括内存资源、文件资源、信号量资源、共享内存资源,或者引用计数减一,或者彻底释放。…...

【LeetCode热题100】【二叉树】二叉树的最大深度
题目链接:104. 二叉树的最大深度 - 力扣(LeetCode) 最大深度等于左子树的最大深度和右子树的最大深度中的较大者加一 class Solution { public:int maxDepth(TreeNode *root) {if (!root)return 0;return max(maxDepth(root->left), max…...

想做产品经理,应该选择什么专业?
产品经理作为互联网公司的核心职位,一直以来备受关注。随着互联网的不断发展,产品经理的需求也越来越高,很多人都想要了解哪些专业适合做产品经理。本文将为大家介绍几个适合做产品经理的专业。 1、心理学相关专业 C端产品工作的本源&#x…...

[机器学习Day 1~3
[机器学习]Day 1~3 数据预处理第1步:导入库第2步:导入数据集第3步:处理丢失数据第4步:解析分类数据创建虚拟变量 第5步:拆分数据集为训练集合和测试集合第6步:特征量化 简单线性回归模型第一步:…...

Day106:代码审计-PHP原生开发篇文件安全上传监控功能定位关键搜索1day挖掘
目录 emlog-文件上传&文件删除 emlog-模板文件上传 emlog-插件文件上传 emlog-任意文件删除 通达OA-文件上传&文件包含 知识点: PHP审计-原生开发-文件上传&文件删除-Emlog PHP审计-原生开发-文件上传&文件包含-通达OA emlog-文件上传&文件…...

数码视讯Q7盒子刷armbian遇到的坑之二
继续,nand的q7 搜遍全网,这个盒子能用的安卓映像有两个,一个本站付费下载的那个,另一个是20191218-Q7-nand-4.4.2-root-twrp-Milton这个映像(具体地址自己搜索吧)。第一个需要license,需要自己…...

vue2 使用vue-org-tree demo
1.安装 npm i vue2-org-tree npm install -D less-loader less安装 less-loader出错解决办法,直接在package.json》devDependencies下面加入less和less-loader版本,然后执行npm i ,我用的nodejs版本是 16.18.0,“webpack”: “^4…...

【数据结构】考研真题攻克与重点知识点剖析 - 第 7 篇:查找
前言 本文基础知识部分来自于b站:分享笔记的好人儿的思维导图与王道考研课程,感谢大佬的开源精神,习题来自老师划的重点以及考研真题。此前我尝试了完全使用Python或是结合大语言模型对考研真题进行数据清洗与可视化分析,本人技术…...

【数仓】DataX 通过SpringBoot项目自动生成 job.json 文件
相关文章 【数仓】基本概念、知识普及、核心技术【数仓】数据分层概念以及相关逻辑【数仓】Hadoop软件安装及使用(集群配置)【数仓】Hadoop集群配置常用参数说明【数仓】zookeeper软件安装及集群配置【数仓】kafka软件安装及集群配置【数仓】flume软件安…...

注解式 WebSocket - 构建 群聊、单聊 系统
目录 前言 注解式 WebSocket 构建聊天系统 群聊系统(基本框架) 群聊系统(添加昵称) 单聊系统 WebSocket 作用域下无法注入 Spring Bean 对象? 考虑离线消息 前言 很久之前,咱们聊过 WebSocket 编程式…...

无线游戏手柄的测试(Windows11系统手柄调试方法)
实物 1、把游戏手柄的无线接收器插入到电脑usb接口中 2、【控制面板】----【查看设备和打印机】 3、【蓝牙和其它设备】--【更多设备和打印机设置】 4、鼠标右键【游戏控制器设置】 5、【属性】 6、【测试】(每个按键是否正常) 7、【校准】(…...

计算机的各种转换
一、存量容量的转换 特别注意:1 B 8 bit 转换为:1024 2(10) 括号中的数字为2的指数(即多少次方) 1KB2(10)B1024B; 括号中的数字为2的指数(即多少次方) 1MB2(10)KB1024KB2(20)B; 1GB2(10)MB1024MB2(3…...

Git分布式版本控制系统——Git常用命令(一)
一、获取Git仓库--在本地初始化仓库 执行步骤如下: 1.在任意目录下创建一个空目录(例如GitRepos)作为我们的本地仓库 2.进入这个目录中,点击右键打开Git bash窗口 3.执行命令git init 如果在当前目录中看到.git文件夹&#x…...

【Node.js】短链接
原文链接:Nodejs 第六十二章(短链接) - 掘金 (juejin.cn) 短链接是一种缩短长网址的方法,将原始的长网址转换为更短的形式。短链接的主要用途之一是在社交媒体平台进行链接分享。由于这些平台对字符数量有限制,长网址可…...

详解 Redis 在 Centos 系统上的安装
文章目录 详解 Redis 在 Centos 系统上的安装1. 使用 yum 安装 Redis 52. 创建符号链接3. 修改配置文件4. 启动和停止 Redis 详解 Redis 在 Centos 系统上的安装 1. 使用 yum 安装 Redis 5 如果是Centos8,yum 仓库中默认的 redis 版本就是5,直接 yum i…...

C语言 | Leetcode C语言题解之第17题电话号码的字母组合
题目: 题解: char phoneMap[11][5] {"\0", "\0", "abc\0", "def\0", "ghi\0", "jkl\0", "mno\0", "pqrs\0", "tuv\0", "wxyz\0"};char* digits…...

wordpress全站开发指南-面向开发者及深度用户(全中文实操)--wordpress中的著名循环
wordpress中的著名循环 首先,在深入研究任何代码之前,我们首先要确保我们有不止一篇博客文章可以工作。因此,我们要去自己的wordpress站点,从侧边栏单机Posts(文章),进行创建 在执行代码的时候会优先执行single.php如…...

libVLC 提取视频帧使用QGraphicsView渲染
在前面章节中,我们讲解了如何使用QWidget渲染每一帧视频数据,这种方法对 CPU 负荷较高。 libVLC 提取视频帧使用QWidget渲染-CSDN博客 后面又讲解了使用OpenGL渲染每一帧视频数据,使用 OpenGL去绘制,利用 GPU 减轻 CPU 计算负荷…...

大厂Java笔试题之判断字母大小写
/*** 题目:如果一个由字母组成的字符串,首字母是大写,那么就统计该字符串中大写字母的数量,并输出该字符串中所有的大写字母。否则,就输出* 该字符串不是首字母大写*/ public class Demo2 {public static void main(St…...

场景文本检测识别学习 day02(AlexNet论文阅读、ResNet论文精读)
怎么读论文 在第一遍阅读的时候,只需要看题目,摘要和结论,先看题目是不是跟我的方向有关,看摘要是不是用到了我感兴趣的方法,看结论他是怎么解决摘要中提出的问题,或者怎么实现摘要中的方法,然…...

4.9日总结
1.MySQL概述 1.数据库基本概念:存储数据的仓库,数据是有组织的进行存储 2.数据库管理系统:操纵和管理数据库的大型软件 3.SQL:操作关系型数据库的编程语言,定义了一套操作型数据库统一标准 2.MySQL数据库 关系型数…...

python第四次作业
1、找出10000以内能被5或6整除,但不能被两者同时整除的数(函数) def func():for i in range(10001):if (i % 5 0 or i % 6 0) and i % 30 ! 0:print(i,end " ")func() 2、写一个方法,计算列表所有偶数下标元素的…...

工业通信原理——Modbus-TCP通信规约定义
工业通信原理——Modbus-TCP通信规约定义 前言 Modbus TCP是一种基于TCP/IP协议的通信规约,用于在客户机和服务器之间进行数据通信。 Modbus-TCP通信规约定义 Modbus TCP通信规约的定义,包括客户机请求和服务器响应的基本流程: 连接建立…...

Vue - 4( 8000 字 Vue 入门级教程)
一: Vue 初阶 1.1 关于不同版本的 Vue Vue.js 有不同版本,如 vue.js 与 vue.runtime.xxx.js,这些版本主要针对不同的使用场景和需求进行了优化,区别主要体现在以下几个方面: 完整版 vs 运行时版: vue.js&…...

5.118 BCC工具之xfsslower.py解读
一,工具简介 xfsslower显示了XFS的读取、写入、打开和fsync操作,这些操作慢于一个阈值。 二,代码示例 #!/usr/bin/env pythonfrom __future__ import print_function from bcc import BPF import argparse from time import strftime# arguments examples = ""…...

Spark编程基础
一、RDD入门 1.RDD是什么? RDD是一个容错的、只读的、可进行并行操作的数据结构,是一个分布在集群各个节点中的存放元素的集合,即弹性分布式数据集。 2.RDD的三种创建方式 第一种是将程序中已存在的集合(如集合、列表、数组&a…...

React 状态管理:高效处理数组数据的5种方法
1.原因 为什么在 React 中,状态(state)如果是数组类型,需要单独处理?主要有以下几个原因: 不可变性(Immutability): React 中的状态是不可变的,意味着我们不能直接修改状态,而是要创建一个新的状态对象。对于数组来说,直接修改数组元素是不符合 React 的设计原则的…...