赞
踩
大家好,今天带大家一起系统的学习下模拟浏览器运行库Selenium
,它是一个用于Web
自动化测试及爬虫应用的重要工具。
Selenium
测试直接运行在浏览器中,就像真正的用户在操作一样。支持的浏览器包括IE
(7, 8, 9, 10, 11),Mozilla Firefox
,Safari
,Google Chrome
,Opera
和Edge
等。
这里我将以Chrome
为例进行Selenium
功能的演示~
目录:
0. 准备工作
0.1. 安装selenium库
0.2. 安装浏览器驱动
1. 基本用法
1.1. 初始化浏览器对象
1.2. 访问页面
1.3. 设置浏览器大小
1.4. 刷新页面
1.5. 前进后退
2. 获取页面基础属性
3. 定位页面元素
3.1. id定位
3.2. name定位
3.3. class定位
3.4. tag定位
3.5. link定位
3.6. partial定位
3.7. xpath定位
3.8. css定位
3.9. find_element的By定位
3.10. 多个元素
4. 获取页面元素属性
4.1. get_attribute获取属性
4.2. 获取文本
4.3. 获取其他属性
5. 页面交互操作
5.1. 输入文本
5.2. 点击
5.3. 清除文本
5.4. 回车确认
5.5. 单选
5.6. 多选
5.7. 下拉框
6. 多窗口切换
6.1. Frame切换
6.2. 选项卡切换
7. 模拟鼠标操作
7.1. 左键
7.2. 右键
7.3. 双击
7.4. 拖拽
7.5. 悬停
8. 模拟键盘操作
9. 延时等待
9.1. 强制等待
9.2. 隐式等待
9.3. 显式等待
10. 其他
10.1. 运行JavaScript
10.2. Cookie
10.3. 反屏蔽
0. 准备工作
在开始后续功能演示之前,我们需要先安装Chrome
浏览器并配置好ChromeDriver
,当然也需要安装selenium
库!
pip install selenium
其实,有两种方式安装浏览器驱动:一种是常见的手动安装,另一种则是利用第三方库自动安装。
以下前提:大家都已经安装好了Chrome
浏览器哈
手动安装
先查看本地Chrome
浏览器版本:(两种方式均可)
在浏览器的地址栏键入Chrome://version
,即可查看浏览器版本号
或者点击Chrome
菜单 帮助→关于Google Chrome,查看浏览器版本号
再选择对应版本号的驱动版本
下载地址:https://chromedriver.storage.googleapis.com/index.html
最后进行环境变量配置,也就是将对应的ChromeDriver
的可执行文件chromedriver.exe
文件拖到Python
的Scripts
目录下。
注:当然也可以不这样做,但是在调用的时候指定chromedriver.exe
绝对路径亦可。
自动安装
自动安装需要用到第三方库webdriver_manager
,先安装这个库,然后调用对应的方法即可。
- from selenium import webdriver
- from selenium.webdriver.common.keys import Keys
- from webdriver_manager.chrome import ChromeDriverManager
-
- browser = webdriver.Chrome(ChromeDriverManager().install())
-
- browser.get('http://www.baidu.com')
- search = browser.find_element_by_id('kw')
- search.send_keys('python')
- search.send_keys(Keys.ENTER)
-
- # 关闭浏览器
- browser.close()
在上述代码中,ChromeDriverManager().install()
方法就是自动安装驱动的操作,它会自动获取当前浏览器的版本并去下载对应的驱动到本地。
- ====== WebDriver manager ======
- Current google-chrome version is 96.0.4664
- Get LATEST chromedriver version for 96.0.4664 google-chrome
- There is no [win32] chromedriver for browser in cache
- Trying to download new driver from https://chromedriver.storage.googleapis.com/96.0.4664.45/chromedriver_win32.zip
- Driver has been saved in cache [C:\Users\Gdc\.wdm\drivers\chromedriver\win32\96.0.4664.45]
如果本地已经有该浏览器渠道,则会提示其已存在。
- ====== WebDriver manager ======
- Current google-chrome version is 96.0.4664
- Get LATEST driver version for 96.0.4664
- Driver [C:\Users\Gdc\.wdm\drivers\chromedriver\win32\96.0.4664.45\chromedriver.exe] found in cache
搞定以上准备工作,我们就可以开始本文正式内容的学习啦~
1. 基本用法
这节我们就从初始化浏览器对象、访问页面、设置浏览器大小、刷新页面和前进后退等基础操作。
在准备工作部分我们提到需要将浏览器渠道添加到环境变量或者指定绝对路径,前者可以直接初始化后者则需要进行指定。
- from selenium import webdriver
-
- # 初始化浏览器为chrome浏览器
- browser = webdriver.Chrome()
-
- # 指定绝对路径的方式
- path = r'C:\Users\Gdc\.wdm\drivers\chromedriver\win32\96.0.4664.45\chromedriver.exe'
- browser = webdriver.Chrome(path)
-
- # 关闭浏览器
- browser.close()
初始化浏览器对象
可以看到以上是有界面的浏览器,我们还可以初始化浏览器为无界面的浏览器。
- from selenium import webdriver
-
- # 无界面的浏览器
- option = webdriver.ChromeOptions()
- option.add_argument("headless")
- browser = webdriver.Chrome(options=option)
-
- # 访问百度首页
- browser.get(r'https://www.baidu.com/')
- # 截图预览
- browser.get_screenshot_as_file('截图.png')
-
- # 关闭浏览器
- browser.close()
截图
完成浏览器对象的初始化后并将其赋值给了browser
对象,接下来我们就可以调用browser
来执行各种方法模拟浏览器的操作了。
进行页面访问使用的是get
方法,传入参数为待访问页面的URL
地址即可。
- from selenium import webdriver
-
- # 初始化浏览器为chrome浏览器
- browser = webdriver.Chrome()
-
- # 访问百度首页
- browser.get(r'https://www.baidu.com/')
-
- # 关闭浏览器
- browser.close()
set_window_size()
方法可以用来设置浏览器大小(就是分辨率),而maximize_window
则是设置浏览器为全屏!
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- # 设置浏览器大小:全屏
- browser.maximize_window()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 设置分辨率 500*500
- browser.set_window_size(500,500)
- time.sleep(2)
-
- # 设置分辨率 1000*800
- browser.set_window_size(1000,800)
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
这里就不截图了,大家自行演示看效果哈~
刷新页面是我们在浏览器操作时很常用的操作,这里refresh()
方法可以用来进行浏览器页面刷新。
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- # 设置浏览器全屏
- browser.maximize_window()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- try:
- # 刷新页面
- browser.refresh()
- print('刷新页面')
- except Exception as e:
- print('刷新失败')
-
- # 关闭浏览器
- browser.close()
大家也是自行演示看效果哈,同F5
快捷键。
前进后退也是我们在使用浏览器时非常常见的操作,这里forward()
方法可以用来实现前进,back()
可以用来实现后退。
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- # 设置浏览器全屏
- browser.maximize_window()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 打开淘宝页面
- browser.get(r'https://www.taobao.com')
- time.sleep(2)
-
- # 后退到百度页面
- browser.back()
- time.sleep(2)
-
- # 前进的淘宝页面
- browser.forward()
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
2. 获取页面基础属性
当我们用selenium
打开某个页面,有一些基础属性如网页标题、网址、浏览器名称、页面源码等信息。
- from selenium import webdriver
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
-
- # 网页标题
- print(browser.title)
- # 当前网址
- print(browser.current_url)
- # 浏览器名称
- print(browser.name)
- # 网页源码
- print(browser.page_source)
输出如下:
- 百度一下,你就知道
- https://www.baidu.com/
- chrome
- <html><head><script async="" src="https://passport.baidu.com/passApi/js/wrapper.js?cdnversion=1640515789507&_=1640515789298"></script><meta http-equiv="Content-Type" content="text/html;charset=utf-8"><meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"><meta content="always" name="referrer"><meta name="theme-color"..."
需要注意的是,这里的页面源码我们就可以用正则表达式
、Bs4
、xpath
以及pyquery
等工具进行解析提取想要的信息了。
3. 定位页面元素
我们在实际使用浏览器的时候,很重要的操作有输入文本、点击确定等等。对此,Selenium
提供了一系列的方法来方便我们实现以上操作。常说的8种
定位页面元素的操作方式,我们一一演示一下!
我们以百度首页的搜索框节点为例,搜索python
搜索框
搜索框的html
结构:
<input id="kw" name="wd" class="s_ipt" value="" maxlength="255" autocomplete="off">
find_element_by_id()
根据id
属性获取,这里id
属性是 kw
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 在搜索框输入 python
- browser.find_element_by_id('kw').send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
find_element_by_name()
根据name
属性获取,这里name
属性是 wd
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 在搜索框输入 python
- browser.find_element_by_name('wd').send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
find_element_by_class_name()
根据class
属性获取,这里class
属性是s_ipt
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 在搜索框输入 python
- browser.find_element_by_class_name('s_ipt').send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
我们知道HTML
是通过tag
来定义功能的,比如input
是输入,table
是表格等等。每个元素其实就是一个tag
,一个tag
往往用来定义一类功能,我们查看百度首页的html
代码,可以看到有很多同类tag
,所以其实很难通过tag
去区分不同的元素。
find_element_by_tag_name()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 在搜索框输入 python
- browser.find_element_by_tag_name('input').send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
由于存在多个input
,以上代码会报错。
这种方法顾名思义就是用来定位文本链接的,比如百度首页上方的分类模块链接。
find_element_by_link_text()
以新闻为例
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 点击新闻 链接
- browser.find_element_by_link_text('新闻').click()
- time.sleep(2)
-
- # 关闭浏览器全部页面
- browser.quit()
有时候一个超链接的文本很长,我们如果全部输入,既麻烦,又显得代码很不美观,这时候我们就可以只截取一部分字符串,用这种方法模糊匹配了。
find_element_by_partial_link_text()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 点击新闻 链接
- browser.find_element_by_partial_link_text('闻').click()
- time.sleep(2)
-
- # 关闭浏览器全部页面
- browser.quit()
前面介绍的几种定位方法都是在理想状态下,有一定使用范围的,那就是:在当前页面中,每个元素都有一个唯一的id
或name
或class
或超链接文本
的属性,那么我们就可以通过这个唯一的属性值来定位他们。
但是在实际工作中并非有这么美好,那么这个时候我们就只能通过xpath
或者css
来定位了。
find_element_by_xpath()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 在搜索框输入 python
- browser.find_element_by_xpath("//*[@id='kw']").send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
这种方法相对xpath
要简洁些,定位速度也要快些。
find_element_by_css_selector()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 在搜索框输入 python
- browser.find_element_by_css_selector('#kw').send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
除了上述的8
种定位方法,Selenium
还提供了一个通用的方法find_element()
,这个方法有两个参数:定位方式和定位值。
- # 使用前先导入By类
- from selenium.webdriver.common.by import By
以上的操作可以等同于以下:
- browser.find_element(By.ID,'kw')
- browser.find_element(By.NAME,'wd')
- browser.find_element(By.CLASS_NAME,'s_ipt')
- browser.find_element(By.TAG_NAME,'input')
- browser.find_element(By.LINK_TEXT,'新闻')
- browser.find_element(By.PARTIAL_LINK_TEXT,'闻')
- browser.find_element(By.XPATH,'//*[@id="kw"]')
- browser.find_element(By.CSS_SELECTOR,'#kw')
如果定位的目标元素在网页中不止一个,那么则需要用到find_elements
,得到的结果会是列表形式。简单来说,就是element
后面多了复数标识s
,其他操作一致。
4. 获取页面元素属性
既然我们有很多方式来定位页面的元素,那么接下来就可以考虑获取以下元素的属性了,尤其是用Selenium
进行网络爬虫的时候。
以百度首页的logo为例,获取logo相关属性
<img hidefocus="true" id="s_lg_img" class="index-logo-src" src="//www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png" width="270" height="129" onerror="this.src='//www.baidu.com/img/flexible/logo/pc/index.png';this.onerror=null;" usemap="#mp">
获取logo的图片地址
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
-
- logo = browser.find_element_by_class_name('index-logo-src')
- print(logo)
- print(logo.get_attribute('src'))
-
- # 关闭浏览器
- browser.close()
输出:
- <selenium.webdriver.remote.webelement.WebElement (session="e95b18c43a330745af019e0041f0a8a4", element="7dad5fc0-610b-45b6-b543-9e725ee6cc5d")>
- https://www.baidu.com/img/PCtm_d9c8750bed0b3c7d089fa7d55720d6cf.png
以热榜为例,获取热榜文本和链接
<a class="title-content tag-width c-link c-font-medium c-line-clamp1" href="https://www.baidu.com/s?cl=3&tn=baidutop10&fr=top1000&wd=%E5%90%84%E5%9C%B0%E8%B4%AF%E5%BD%BB%E5%8D%81%E4%B9%9D%E5%B1%8A%E5%85%AD%E4%B8%AD%E5%85%A8%E4%BC%9A%E7%B2%BE%E7%A5%9E%E7%BA%AA%E5%AE%9E&rsv_idx=2&rsv_dl=fyb_n_homepage&sa=fyb_n_homepage&hisfilter=1" target="_blank"><span class="title-content-index c-index-single c-index-single-hot1">1</span><span class="title-content-title">各地贯彻十九届六中全会精神纪实</span></a>
获取热榜的文本,用的是text
属性,直接调用即可
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
-
- logo = browser.find_element_by_css_selector('#hotsearch-content-wrapper > li:nth-child(1) > a')
- print(logo.text)
- print(logo.get_attribute('href'))
-
- # 关闭浏览器
- browser.close()
输出:
- 1各地贯彻十九届六中全会精神纪实
- https://www.baidu.com/s?cl=3&tn=baidutop10&fr=top1000&wd=%E5%90%84%E5%9C%B0%E8%B4%AF%E5%BD%BB%E5%8D%81%E4%B9%9D%E5%B1%8A%E5%85%AD%E4%B8%AD%E5%85%A8%E4%BC%9A%E7%B2%BE%E7%A5%9E%E7%BA%AA%E5%AE%9E&rsv_idx=2&rsv_dl=fyb_n_homepage&sa=fyb_n_homepage&hisfilter=1
除了属性和文本值外,还有id、位置、标签名和大小等属性。
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- browser.get(r'https://www.baidu.com')
-
- logo = browser.find_element_by_class_name('index-logo-src')
- print(logo.id)
- print(logo.location)
- print(logo.tag_name)
- print(logo.size)
-
- # 关闭浏览器
- browser.close()
输出:
- 6af39c9b-70e8-4033-8a74-7201ae09d540
- {'x': 490, 'y': 46}
- img
- {'height': 129, 'width': 270}
5. 页面交互操作
页面交互就是在浏览器的各种操作,比如上面演示过的输入文本、点击链接等等,还有像清除文本、回车确认、单选框与多选框选中等。
其实,在之前的小节中我们有用过此操作。
send_keys()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 定位搜索框
- input = browser.find_element_by_class_name('s_ipt')
- # 输入python
- input.send_keys('python')
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
同样,我们也用过这个点击操作。
click()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 选中新闻按钮
- click = browser.find_element_by_link_text('新闻')
- # 点击之
- click.click()
- time.sleep(2)
-
- # 关闭浏览器全部页面
- browser.quit()
既然有输入,这里也就有清除文本啦。
clear()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 定位搜索框
- input = browser.find_element_by_class_name('s_ipt')
- # 输入python
- input.send_keys('python')
- time.sleep(2)
- # 清除python
- input.clear()
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
比如,在搜索框输入文本python
,然后回车就出查询操作结果的情况。
submit()
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 定位搜索框
- input = browser.find_element_by_class_name('s_ipt')
- # 输入python
- input.send_keys('python')
- time.sleep(2)
- # 回车查询
- input.submit()
- time.sleep(5)
-
- # 关闭浏览器
- browser.close()
单选比较好操作,先定位需要单选的某个元素,然后点击一下即可。
多选好像也比较容易,依次定位需要选择的元素,点击即可。
下拉框的操作相对复杂一些,需要用到Select
模块。
先导入该类
from selenium.webdriver.support.select import Select
在select
模块中有以下定位方法
- '''1、三种选择某一选项项的方法'''
-
- select_by_index() # 通过索引定位;注意:index索引是从“0”开始。
- select_by_value() # 通过value值定位,value标签的属性值。
- select_by_visible_text() # 通过文本值定位,即显示在下拉框的值。
-
- '''2、三种返回options信息的方法'''
-
- options # 返回select元素所有的options
- all_selected_options # 返回select元素中所有已选中的选项
- first_selected_options # 返回select元素中选中的第一个选项
-
-
- '''3、四种取消选中项的方法'''
-
- deselect_all # 取消全部的已选择项
- deselect_by_index # 取消已选中的索引项
- deselect_by_value # 取消已选中的value值
- deselect_by_visible_text # 取消已选中的文本值
我们来进行演示一波,由于暂时没找到合适的网页,我这边写了一个简单的网页本地测试(文件存为 帅哥.html)
- <html>
-
- <body>
-
- <form>
- <select name="帅哥">
- <option value="才哥">才哥</option>
- <option value="小明" selected="">小明</option>
- <option value="小华">小华</option>
- <option value="草儿">小草</option>
- </select>
- </form>
-
- </body>
- </html>
然后,再演示下拉框的不同选择的方式
- from selenium import webdriver
- from selenium.webdriver.support.select import Select
- import time
-
- url = 'file:///C:/Users/Gdc/Desktop/帅哥.html'
-
- browser = webdriver.Chrome()
-
- browser.get(url)
- time.sleep(2)
-
- # 根据索引选择
- Select(browser.find_element_by_name("帅哥")).select_by_index("2")
- time.sleep(2)
- # 根据value值选择
- Select(browser.find_element_by_name("帅哥")).select_by_value("草儿")
- time.sleep(2)
- # 根据文本值选择
- Select(browser.find_element_by_name("帅哥")).select_by_visible_text("才哥")
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
下拉框
6. 多窗口切换
比如同一个页面的不同子页面的节点元素获取操作,不同选项卡之间的切换以及不同浏览器窗口之间的切换操作等等。
Selenium
打开一个页面之后,默认是在父页面进行操作,此时如果这个页面还有子页面,想要获取子页面的节点元素信息则需要切换到子页面进行擦走,这时候switch_to.frame()
就来了。如果想回到父页面,用switch_to.parent_frame()
即可。
我们在访问网页的时候会打开很多个页面,在Selenium
中提供了一些方法方便我们对这些页面进行操作。
current_window_handle
:获取当前窗口的句柄。
window_handles
:返回当前浏览器的所有窗口的句柄。
switch_to_window()
:用于切换到对应的窗口。
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
-
- # 打开百度
- browser.get('http://www.baidu.com')
- # 新建一个选项卡
- browser.execute_script('window.open()')
- print(browser.window_handles)
- # 跳转到第二个选项卡并打开知乎
- browser.switch_to.window(browser.window_handles[1])
- browser.get('http://www.zhihu.com')
- # 回到第一个选项卡并打开淘宝(原来的百度页面改为了淘宝)
- time.sleep(2)
- browser.switch_to.window(browser.window_handles[0])
- browser.get('http://www.taobao.com')
7. 模拟鼠标操作
既然是模拟浏览器操作,自然也就需要能模拟鼠标的一些操作了,这里需要导入ActionChains
类。
from selenium.webdriver.common.action_chains import ActionChains
这个其实就是页面交互操作中的点击click()
操作。
context_click()
- from selenium.webdriver.common.action_chains import ActionChains
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 定位到要右击的元素,这里选的新闻链接
- right_click = browser.find_element_by_link_text('新闻')
-
- # 执行鼠标右键操作
- ActionChains(browser).context_click(right_click).perform()
- time.sleep(2)
-
- # 关闭浏览器
- browser.close()
在上述操作中
ActionChains(browser)
:调用ActionChains()
类,并将浏览器驱动browser
作为参数传入
context_click(right_click)
:模拟鼠标双击,需要传入指定元素定位作为参数
perform()
:执行ActionChains()
中储存的所有操作,可以看做是执行之前一系列的操作
double_click()
- from selenium.webdriver.common.action_chains import ActionChains
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- browser.get(r'https://www.baidu.com')
- time.sleep(2)
-
- # 定位到要双击的元素
- double_click = browser.find_element_by_css_selector('#bottom_layer > div > p:nth-child(8) > span')
-
- # 双击
- ActionChains(browser).double_click(double_click).perform()
- time.sleep(15)
-
- # 关闭浏览器
- browser.close()
drag_and_drop(source,target)
拖拽操作嘛,开始位置和结束位置需要被指定,这个常用于滑块类验证码的操作之类。
我们以菜鸟教程的一个案例来进行演示
https://www.runoob.com/try/try.php?filename=jqueryui-api-droppable
- from selenium.webdriver.common.action_chains import ActionChains
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- url = 'https://www.runoob.com/try/try.php?filename=jqueryui-api-droppable'
- browser.get(url)
- time.sleep(2)
-
- browser.switch_to.frame('iframeResult')
-
- # 开始位置
- source = browser.find_element_by_css_selector("#draggable")
-
- # 结束位置
- target = browser.find_element_by_css_selector("#droppable")
-
- # 执行元素的拖放操作
- actions = ActionChains(browser)
- actions.drag_and_drop(source, target)
- actions.perform()
- # 拖拽
- time.sleep(15)
-
- # 关闭浏览器
- browser.close()
拖拽
move_to_element()
- from selenium.webdriver.common.action_chains import ActionChains
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- url = 'https://www.baidu.com'
- browser.get(url)
- time.sleep(2)
-
- # 定位悬停的位置
- move = browser.find_element_by_css_selector("#form > span.bg.s_ipt_wr.new-pmd.quickdelete-wrap > span.soutu-btn")
-
- # 悬停操作
- ActionChains(browser).move_to_element(move).perform()
- time.sleep(5)
-
- # 关闭浏览器
- browser.close()
悬停效果
8. 模拟键盘操作
selenium
中的Keys()
类提供了大部分的键盘操作方法,通过send_keys()
方法来模拟键盘上的按键。
引入Keys
类
from selenium.webdriver.common.keys import Keys
常见的键盘操作
send_keys(Keys.BACK_SPACE)
:删除键(BackSpace)
send_keys(Keys.SPACE)
:空格键(Space)
send_keys(Keys.TAB)
:制表键(TAB)
send_keys(Keys.ESCAPE)
:回退键(ESCAPE)
send_keys(Keys.ENTER)
:回车键(ENTER)
send_keys(Keys.CONTRL,'a')
:全选(Ctrl+A)
send_keys(Keys.CONTRL,'c')
:复制(Ctrl+C)
send_keys(Keys.CONTRL,'x')
:剪切(Ctrl+X)
send_keys(Keys.CONTRL,'v')
:粘贴(Ctrl+V)
send_keys(Keys.F1)
:键盘F1.....
send_keys(Keys.F12)
:键盘F12
实例操作演示:
定位需要操作的元素,然后操作即可!
- from selenium.webdriver.common.keys import Keys
- from selenium import webdriver
- import time
-
- browser = webdriver.Chrome()
- url = 'https://www.baidu.com'
- browser.get(url)
- time.sleep(2)
-
- # 定位搜索框
- input = browser.find_element_by_class_name('s_ipt')
- # 输入python
- input.send_keys('python')
- time.sleep(2)
-
- # 回车
- input.send_keys(Keys.ENTER)
- time.sleep(5)
-
- # 关闭浏览器
- browser.close()
9. 延时等待
如果遇到使用ajax
加载的网页,页面元素可能不是同时加载出来的,这个时候尝试在get
方法执行完成时获取网页源代码可能并非浏览器完全加载完成的页面。所以,这种情况下需要设置延时等待一定时间,确保全部节点都加载出来。
三种方式可以来玩玩:强制等待、隐式等待和显式等待
就很简单了,直接time.sleep(n)
强制等待n秒,在执行get
方法之后执行。
implicitly_wait()
设置等待时间,如果到时间有元素节点没有加载出来,就会抛出异常。
- from selenium import webdriver
-
- browser = webdriver.Chrome()
-
- # 隐式等待,等待时间10秒
- browser.implicitly_wait(10)
-
- browser.get('https://www.baidu.com')
- print(browser.current_url)
- print(browser.title)
-
- # 关闭浏览器
- browser.close()
设置一个等待时间和一个条件,在规定时间内,每隔一段时间查看下条件是否成立,如果成立那么程序就继续执行,否则就抛出一个超时异常。
- from selenium import webdriver
- from selenium.webdriver.support.wait import WebDriverWait
- from selenium.webdriver.support import expected_conditions as EC
- from selenium.webdriver.common.by import By
- import time
-
- browser = webdriver.Chrome()
- browser.get('https://www.baidu.com')
- # 设置等待时间10s
- wait = WebDriverWait(browser, 10)
- # 设置判断条件:等待id='kw'的元素加载完成
- input = wait.until(EC.presence_of_element_located((By.ID, 'kw')))
- # 在关键词输入:关键词
- input.send_keys('Python')
-
- # 关闭浏览器
- time.sleep(2)
- browser.close()
WebDriverWait的参数说明:
WebDriverWait(driver,timeout,poll_frequency=0.5,ignored_exceptions=None)
driver
: 浏览器驱动
timeout
: 超时时间,等待的最长时间(同时要考虑隐性等待时间)
poll_frequency
: 每次检测的间隔时间,默认是0.5秒
ignored_exceptions
:超时后的异常信息,默认情况下抛出NoSuchElementException
异常until(method,message='')
method
: 在等待期间,每隔一段时间调用这个传入的方法,直到返回值不是False
message
: 如果超时,抛出TimeoutException
,将message
传入异常until_not(method,message='')
until_not
与until
相反,until
是当某元素出现或什么条件成立则继续执行,until_not
是当某元素消失或什么条件不成立则继续执行,参数也相同。
其他等待条件
- from selenium.webdriver.support import expected_conditions as EC
-
- # 判断标题是否和预期的一致
- title_is
- # 判断标题中是否包含预期的字符串
- title_contains
-
- # 判断指定元素是否加载出来
- presence_of_element_located
- # 判断所有元素是否加载完成
- presence_of_all_elements_located
-
- # 判断某个元素是否可见. 可见代表元素非隐藏,并且元素的宽和高都不等于0,传入参数是元组类型的locator
- visibility_of_element_located
- # 判断元素是否可见,传入参数是定位后的元素WebElement
- visibility_of
- # 判断某个元素是否不可见,或是否不存在于DOM树
- invisibility_of_element_located
-
- # 判断元素的 text 是否包含预期字符串
- text_to_be_present_in_element
- # 判断元素的 value 是否包含预期字符串
- text_to_be_present_in_element_value
-
- #判断frame是否可切入,可传入locator元组或者直接传入定位方式:id、name、index或WebElement
- frame_to_be_available_and_switch_to_it
-
- #判断是否有alert出现
- alert_is_present
-
- #判断元素是否可点击
- element_to_be_clickable
-
- # 判断元素是否被选中,一般用在下拉列表,传入WebElement对象
- element_to_be_selected
- # 判断元素是否被选中
- element_located_to_be_selected
- # 判断元素的选中状态是否和预期一致,传入参数:定位后的元素,相等返回True,否则返回False
- element_selection_state_to_be
- # 判断元素的选中状态是否和预期一致,传入参数:元素的定位,相等返回True,否则返回False
- element_located_selection_state_to_be
-
- #判断一个元素是否仍在DOM中,传入WebElement对象,可以判断页面是否刷新了
- staleness_of
10. 其他
补充一些
还有一些操作,比如下拉进度条,模拟javaScript
,使用execute_script
方法来实现。
- from selenium import webdriver
-
- browser = webdriver.Chrome()
- # 知乎发现页
- browser.get('https://www.zhihu.com/explore')
-
- browser.execute_script('window.scrollTo(0, document.body.scrollHeight)')
- browser.execute_script('alert("To Bottom")')
在selenium
使用过程中,还可以很方便对Cookie
进行获取、添加与删除等操作。
- from selenium import webdriver
-
- browser = webdriver.Chrome()
- # 知乎发现页
- browser.get('https://www.zhihu.com/explore')
- # 获取cookie
- print(f'Cookies的值:{browser.get_cookies()}')
- # 添加cookie
- browser.add_cookie({'name':'才哥', 'value':'帅哥'})
- print(f'添加后Cookies的值:{browser.get_cookies()}')
- # 删除cookie
- browser.delete_all_cookies()
- print(f'删除后Cookies的值:{browser.get_cookies()}')
输出:
- Cookies的值:[{'domain': '.zhihu.com', 'httpOnly': False, 'name': 'Hm_lpvt_98beee57fd2ef70ccdd5ca52b9740c49', 'path': '/', 'secure': False, 'value': '1640537860'}, {'domain': '.zhihu.com', ...]
- 添加后Cookies的值:[{'domain': 'www.zhihu.com', 'httpOnly': False, 'name': '才哥', 'path': '/', 'secure': True, 'value': '帅哥'}, {'domain': '.zhihu.com', 'httpOnly': False, 'name': 'Hm_lpvt_98beee57fd2ef70ccdd5ca52b9740c49', 'path': '/', 'secure': False, 'value': '1640537860'}, {'domain': '.zhihu.com',...]
- 删除后Cookies的值:[]
发现美团直接给Selenium
给屏蔽了,不知道怎么搞!!
以上就是本次全部内容,如果觉得有帮助,还请点赞+在看呀!
下面是配套资料,对于做【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!
最后: 可以在公众号:程序员小濠 ! 免费领取一份216页软件测试工程师面试宝典文档资料。以及相对应的视频学习教程免费分享!,其中包括了有基础知识、Linux必备、Shell、互联网程序原理、Mysql数据库、抓包工具专题、接口测试工具、测试进阶-Python编程、Web自动化测试、APP自动化测试、接口自动化测试、测试高级持续集成、测试架构开发测试框架、性能测试、安全测试等。
如果我的博客对你有帮助、如果你喜欢我的博客内容,请 “点赞” “评论” “收藏” 一键三连哦!喜欢软件测试的小伙伴们,可以加入我们的测试技术交流扣扣群:779450660里面有各种软件测试资源和技术讨论)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。