3.8 浏览器 copy selector
我们选择元素有一个方法可以快速生成,右键点击元素点击检查定位到元素后,右键点击元素有个 Copy,选择 Copy selector
也会把选择的表达式选择下来 Copy selector 是 Css 的写法,我们看到也有XPath 大家也可以试一下,这种方法其实是非常有限的,这种方式是以一个有 id 的元素为瞄点往下找,如果有 id 的元素不是本身而是上层节点他就要依赖上层节点和他的相对关系来找,如果上层节点跟他隔得很远,他也会一层层的找打他有 id 的上层节点,路径会很长如果他中间的任何一个环节发生变化就容易出问题健壮性不好,他不会根据别的方法比如 class ,这种方法我们了解一下就好很鸡肋。
3.8.1异常捕获,确保 chrom 进程退出
我们做练习的时候不可能一次就把作业做出来,总归要做一些调试出错调试出错,肯定遇到过出错浏览器就没法关闭的状况,有的同学会想不关就不关呗我们手动关上就好了,调试的时候是可以的,假如我们做系统自动化的时候有很多的用例上百个 web 自动化用例在执行,如果遇到报错浏览器没有关闭,第二天我们去看的话机器就像卡死了一样,因为很多用例遇到报错比如找不到元素的错误导致浏览器开了二三十个在那甚至更多,我们可以把自动化逻辑放在异常捕获 try 里面`,看下面例子:
from selenium import webdriver
import time
driver = webdriver.Chrome('E:\ChromDriver\chromdriver2.43\chromedriver.exe')
driver.implicitly_wait(10)
driver.get('https://www.baidu.com/')
print(driver.title)
driver.find_element_by_id('kk').send_keys('宋曲')
driver.quit()
百度的输入框 id 为 kw 我这里改为 kk 这时候,由于不能执行到 driver.quit,浏览器就不能自动关闭,解决方法:
import traceback
from selenium import webdriver
import time
driver = webdriver.Chrome('E:\ChromDriver\chromdriver2.43\chromedriver.exe')
driver.implicitly_wait(10)
try:
driver.get('https://www.baidu.com/')
print(driver.title)
driver.find_element_by_id('kk').send_keys('宋曲')
except:
print(traceback.format_exc())
finally:
driver.quit()
写在 try 里面,finally 后面需要 driver.quit()
3.8.2 界面悬停
我们操作界面的时候不需要点击他,把鼠标移动到上面就会导致页面的变化,比如我们打开华为官网https://www.vmall.com/
鼠标放在上面元素就会弹出出来,大家可能发现我们这里把鼠标放在上面直接右键查看元素也是可以定位到元素的,其实他本身也在 HTML 当中即使光标不放在上面,我们能不能不模拟光标放在上面,可以找到直接去选择元素(笔记本电脑之类的)吗?像这些界面元素如果没有把光标停在上面让界面显示出来,通常是不能操作它的会出错,因为当前这个元素如果处于没有显现出来的状态你就想试着去操作它。
其实这个元素有个属性。class 等于 none 的这种属性,如果有的同学知道一些 css 一些基础知识。可能就是知道如果有个 none 在这
它的 display 是 none。
display元素是 none 的元素意思是什么意思呢,他当前在界面上放在 HTML 里面但是他不显示出来不显示在界面上,Selenium 是高度模拟用户点击操作的,因为他没有显示出来这个时候你点击他会出错的,我们可以把它冻结。
这个时候就不会消失了,这个时候可以去查看内容了。
selenium 怎么模拟这个移动鼠标到某个元素的动作呢?通过 ActionChains 类,ActionChains 类 里面提供了 一些特殊的动作, 比如移动鼠标到某个元素,就是其中之一,ActionChains 本身有个 ActionChains 模块文件本身实现了一个 ActionChains 类,我们查看这个类的定义。
首先代码里写上这个 ActionChains ,我们按住 Ctrl + 鼠标 左键 点击它。就会到它定义的这个模块里面。
这里可以看到 ActionChains 本身有一个 action_chains.py 模块文件,它里面实现了一个 ActionChains 这个类,我们要看这个类比较好的方法,就是 View --> Tool Windows Structure。
就可以看到 ActionChains 这个类实现的方法有哪些,这些红色 m 开头的都是它的方法。
click 点击,click_and_hold 点击按住不动不要松,context_click 右键点击,double_click 双击等。这里不一一介绍大家自己去研究,我们这里用到的就是 move_to_element 悬停就使用它。
Moving the mouse to the middle of an element.
意思是把鼠标放在一个元素的正中心上去。通常悬停就是用它,移动但不按下来。知道了这个方法我们该怎么使用呢?
这里我们能不能 ActionChains.move_to_element 去调用这个方法,这里是不行的,因为这个是实例方法,凡是这种带self
参数的都是实例方法(某个实例的方法,带self就需要传入实例参数的,就是类的实例)。如果调用这个方法你根据类名调用的时候,你一定要实例化,因为有个实例对象要传进去,所以不能直接根据类名去调用它,根据类名直接去调用的通常都是静态方法(一般带修饰符的比如@stationmehtod)。所以这里我们要用到 ActionChains 类,首先要创建一个 ActionChains 的实例对象。知道之后我们就可以使用它了,首先我们要导入这个类,我们首先看一下这个类的目录(鼠标放在模块名上):
前面这个E盘
什么的这是标准库的安装路径不管它,我们从selenium\webdriver\common
看起。
导入:
from selenium.webdriver.common.action_chains import ActionChains
示例代码:
from selenium import webdriver
import time
executable_path = r"d:\tools\webdrivers\chromedriver.exe"
driver = webdriver.Chrome(executable_path)
driver.get('https://www.vmall.com/')
# ---------------------------------------
# 导入ActionChains这个类
from selenium.webdriver.common.action_chains import ActionChains
# 创建实例对象
ac = ActionChains(driver)
#移动不点击 .perform()执行
ac.move_to_element(driver.find_element_by_id('zxnav_1')).perform()
ele= driver.find_element_by_css_selector(
'#zxnav_1 > div.category-panels.relative > ul > li:nth-child(1) > a')
ele.click()
# ---------------------------------------
input('...')
driver.quit()
ActionChains 实例化的时候要接收一个参数,就是 webdriver 就是我们前面创建的 webdriver 对象 driver = webdriver.Chrome(executable_path) 它不是存在变量里面吗,把它作为参数放进去。
ac = ActionChains(driver)
这样就创建了一个ActionChains 对象,然后调用它的 move_to_element 方法,就执行了移动到某个元素上但不点击它的这个操作。
ac.move_to_element(driver.find_element_by_id('zxnav_1')).perform()
这个 move_to_element 是有参数的,参数就是你要移动的这个元素它所对应的 webelement 对象。
那我们怎么创建这个 webelement 对象呢?当然通过我们之前 讲的 find,因为 find 某个元素返回的就是对应这个元素的 webelement 对象。比如说我们这边,它是有 id 的。
move_to_element 只是在 ActionChains 里面声明你要做的动作,最后一定要再调用一个perform()
,move_to_element(driver.find_element_by_id('zxnav_1'))
注意这个东西返回的还是一个 ActionChains 的对象,最后一个 perform()
才是执行这个行为,前面相当于只是定义了我这个动作链里有这个一个动作,移动到某个元素上面去最后要通过一个 perform去执行它。