本章将介绍使用Selenium和PhantomJS两种工具用来加载动态数据,更多内容请参考:Python学习指南css
Selenium是一个Web的自动化测试工具,最初是为网站自动化测试而开发的,最初是为网站自动化测试而开发的,类型像咱们玩游戏用的按键精灵,能够按指定的命令自动化操做,不一样是Selenium能够直接运行在浏览器上,它支持全部主流的浏览器(包括PhantomJS这些无界面的浏览器)。html
Selenium能够根据咱们的指令,让浏览器自动加载页面,获取须要的页面,甚至页面截屏,或者判断网站上某些动做是否发生。python
Selenium本身不带浏览器,不支持浏览器的功能,它须要与第三方浏览器结合在一块儿才能使用。可是咱们有时候须要让它内嵌在代码中运行,全部咱们而已用一个叫PhantomJS的工具代替真实的浏览器。web
能够从PyPI网站下载Selenium库
http://pypi.python.org/simple/sulenium
,也能够用第三方管理器pip命令安装:pip install selenium
Selenium官方参考文档:http://selenium-python.readthedocs.io/index.html
浏览器
PhantomJS
是一个基于Webkit的"无界面"(headless)浏览器,它会把网站加载到内存并执行页面上的JavaScript,由于不会展现图形界面,因此运行起来比完整的浏览器更高效。cookie
若是咱们把Selenium和PhantomJS结合在一块儿,就能够运行一个很是强大的网络爬虫了,这个爬虫能够处理JavaScript、Cookie、headers,以及任何咱们真实用户须要作的事情。网络
注意:PhantomJS只能从它的网站(
http://phantomjs.org/download.html
)下载。由于PhantomJS是一个功能完善(虽然无界面)的浏览器而非一个Python库,因此它不须要像Python的其它库同样安装,但咱们能够经过Selenium调用PhantomJS来直接使用
PhantomsJS官方才考文档:http://phantomjs.org/documention
app
Selenium库里有一个叫WebDriver的API。WebDriver能够控制浏览器的操做,它能够像BeautifulSoup或者其它Selector对象同样用来查找页面元素,与页面上的元素进行交互(发送文本、点击等),以及执行其余动做来运行网络爬虫。less
#-*- coding:utf-8 -*- #主要用来测试selenium使用phantomJs #导入webdriver from selenium import webdriver import time #要想调用键盘按键操做须要引入keys包 from selenium.webdriver.common.keys import Keys #调用环境变量指定的PhantomJS浏览器建立浏览器对象 driver = webdriver.PhantomJS() driver.set_window_size(1366, 768) #若是没有在环境变量指定PhantomJS位置 #driver = webdriver.PhantomJS(executable_path = "./phantomjs") #get方法会一直等到页面加载,而后才会继续程序,一般测试会在这里选择time.sleep(2) driver.get("http://www.baidu.com/") #获取页面名为wraper的id标签的文本内容 data = driver.find_element_by_id('wrapper').text #打印数据内容 print(data) print driver.title #生成页面快照并保存 driver.save_screenshot("baidu.png") # id="kw"是百度搜索输入框,输入字符串"长城" driver.find_element_by_id('kw').send_keys(u'长城') # id="su"是百度搜索按钮,click()是模拟点击 driver.find_element_by_id('su').click() #获取新的页面快照 driver.save_screenshot("长城.png") #打印网页渲染后的源代码 print(driver.page_source) #获取当前页面Cookie print(driver.get_cookies()) #ctrl+a全选输入框内容 driver.find_element_by_id('kw').send_keys(Keys.CONTROL, 'a') #ctrl+x剪切输入框内容 driver.find_element_by_id('kw').send_keys(Keys.CONTROL, 'x') #输入框从新输入内容 driver.find_element_by_id('kw').send_keys('itcast') #模拟Enter回车键 driver.find_element_by_id('su').send_keys(Keys.RETURN) time.sleep(5) #清空输入框内容 driver.find_element_by_id('kw').clear() #生成新的页面快照 driver.save_screenshot('itcast.png') #获取当前url print(driver.current_url) driver.quit()
Selenium的WebDriver提供了各类方法来寻找元素,假设下面有一个表单输入框:dom
<input type="text" name="user-name" id="passwd-id" />
那么:
#获取id标签值 element = driver.find_element_by_id("passwd-id") #获取name值 element = driver.find_element_by_name("user-name") #获取标签名 element = driver.find_element_by_tag("input") #也能够经过XPath来匹配 element = driver.find_element_by_xpath(//input[@id="passwd-id"])
关于元素的选取,有以下的API单个元素选取
- find_element_by_id
- find_element_by_name
- find_element_by_xpath
- find_element_by_link_text
- find_element_by_partial_link_text
- find_eelement_by_tag_name
- find_element_by_class_name
- find_element_by_css_selector
<div id="coolesWidgeEvah">...</div>
element = driver.find_element_by_id("coolesWidgetEvah") ---------or------- from selenium.webdriver.common.by import By element = driver.find_element(by=By.ID, value="coolesWidgetEvah")
<div class="cheese"><span>Cheddar</span></div><div class="cheese"><span>Gouda</span></div>
cheese = driver.find_element_by_class_name('cheese') ----------------or-------------------- from selenium.webdriver.common.by import By cheese = driver.find_elements(By.CLASS_NAME, "cheese")
<iframe src="..."></iframe>
frame = driver.find_element_by_tag_name("iframe") ------------------------or--------------- from selenium.webdriver.common.by import By frame = driver.find_element(By.TAG_NAME, "iframe")
<imput name="cheese" type="text" />
cheese = driver.find_element_by_name("cheese") -------------or------------------------- from selenium.webdrier.common.by import By cheese = driver.find_element(By.NAME, "cheese")
<a href="http://www.google.com/search?q=cheese">cheese</a>
cheese = driver.find_element_by_link_text("cheese") ---------------------or----------------------- from selenium.webdriver.common.by import By cheese = driver.find_element(By.LINK_TEXT, "cheese")
<a href="http://www.google.com/search?q=cheese">search for cheese</a>
cheese = driver.find_element_by_partial_link_text("cheese") -----------------or----------------- from selenium.webdriver.common.by import By cheese = driver.find_element(By.PARTIAL_LINK_TEXT, "cheese")
<div id="food"><span class="dairy">milk</span><span class="dairy">cheese</span></div>
cheese = driver.find_element_by_css_selector("#food span.dairy.aged") ----------------or------------------------------- from selenium.webdriver.common.by import By cheese = driver.find_element(By.CSS_SELECTOR, "#food span.dairy.aged")
<input type="text" name="example" / > <input type="text" name="other" / >
inputs = driver.find_elements_by_xpath("input") -------------------or------------------ from selenium.webdriver.common.by import By inputs = driver.find_elements(By.XPATH, "//input")
有些时候,咱们须要在页面上模拟一些鼠标操做,好比双击、右击、拖拽甚至按住不动等,咱们可经过导入ActionChains类来作到:
示例:
#导入ActionChains类 from selenium.webdrive import ActionChains #鼠标移动到ac位置 ac = driver.find_elenemt_by_xpath('element') ActionChains(driver).move_to_element(ac).perform() #在ac位置单击 ac = driver.find_element_by_xpath('elementA') ActionChains(driver).move_to_element(ac).click(ac).perform() #在ac位置双击 ac = driver.find_element_by_xpath("elementB") ActionChains(driver).move_to_element(ac).double_click(ac).perform() #在ac位置右击 ac = driver.find_element_by_xpath('elementC') ActionChains(driver).move_to_element(ac).context_click(ac).perform() #在ac位置左键单击hold住 ac = driver.find_element_by_xpath('elementF') ActionChains(driver).move_to_element(ac).click_and_hold(ac).perform() #将ac1拖拽到ac2位置 ac1 = driver.find_element_by_xpath('elementD') ac2 = driver.find_element_by_xpath('elementE') ActionChains(driver).drag_and_drop(ac1, ac2).perform()
咱们已经知道了怎样向文本框输入文字,可是有时候咱们会碰到<select></select>
标签的下拉框。直接点击下拉框中的选项不必定可行。
<select id="status" class="form-control valid" onchange="" name = 'status'> <option value=""></option> <option value="0">未审核</option> <option value="1">初审经过</option> <option value="2">复审经过</option> <option value="3">审核不经过</option> </select>
Selenium专门提供了Select类来处理下拉框。其实WebDriver中提供了一个叫Select的方法,能够帮助咱们完成这些事情:
#导入Select类 from selenium.webdriver.support.ui import Select #找到name的选项卡 select = Select(driver.find_element_by_name('status')) #s select.select_by_index(1) select.select_by_value("0") select.select_by_visible_text(u'未审核')
以上是三种选择下拉框的方式,它能够根据索引来选择,能够根据值来选择,能够根据文字来选择。注意:
- index索引从0开始
- value是option标签的一个属性值,并非显示在下拉框中的值
- visible_text实在option标签文本的值,是显示在下拉框的值
所有取消选择怎么办呢?很简单:
select.deselect_all()
当你触发了某个事件以后,页面出现了弹窗提示,处理这个提示或者获取提示信息方法以下:
alert = driver.switch_to_alert()
一个浏览器确定会有不少窗口,因此咱们确定要有方法来实现窗口的切换,切换窗口的方法以下:
driver.switch_to_window('this is window name')
也可使用window_handles方法来获取每一个窗口的操做对象。例如:
for handle in driver.window_handles: driver.switch_to_window(handle)
操做页面的前进和后退功能:
driver.forward() #前进 driver.back() #后退
获取页面每一个Cookies值,用法以下:
for cookie in driver.get_cookies(): print("%s -> %s"%(cookie['name'], cookie['value']))
删除Cookies,用法以下:
#By name driver.delete_cookie('CookieName') #all driver.delete_all_cookies()
注意:这是很是重要的一部分!
如今的网页原来越多采用了Ajax技术,这样程序变不能肯定什么时候某个元素彻底加载出来了。若是实际页面等待事件过长导出某个dom元素还没出来,可是你的代码直接使用了这个WebElement,那么就会抛出NullPointer的异常。
为了不这种元素定位困难并且会提升产生ElementNotVisibleException的几率。因此Selenium提供了两种等待方式,一种是隐式等待,一种是显式等待。
隐式等待就是等待特定的时间,显示等待是指定某一条件知道这个条件成立时继续执行。
显示等待指定了某个条件,而后设置最长等待事件。若是在这个时间还找到没有元素,那么便会抛出异常。
from selenium import webdriver from selenium.webdriver.common.by import By #WebDriverWait库,负责循环等待 from selenium.webdriver.support.ui import WebDriverWait #expected_conditions类,负责条件触发 from selenium.webdriver.support import expected_conditions as EC driver = webdriver.Chrome() driver.get("http://www.xxxx.com/loading") try: #页面一直循环,知道id="myDynamicElement"出现 element = WebDriverWait(driver, 10).until( EC.presence_of_element_located((By.ID, "myDynamicElement")) ) finally: driver.quit()
若是不写参数,程序默认会0.5s调用一次来来查看安苏是否已经生成,若是原本元素时存在的,那么会当即返回。
下面是一些内置的等待条件,你能够直接调用这些条件,而不用本身写某些等待条件了。
title_is title_contains presence_of_element_located visibility_of_element_located visibility_of presence_of_all_elements_located text_to_be_present_in_element text_to_be_present_in_element_value frame_to_be_available_and_switch_to_it invisibility_of_element_located element_to_be_clickable - it is Displayed and Enabled staleness_of element_to_be_selected element_located_to_be_selected element_selection_state_to_be element_located_selection_state_to_be aert_is_present
隐式等待比较简单,就是简单地设置一个等待时间,单位为秒。
from selenium import webdriver driver = webdriver.Chrome() driver.implicitly_wait(10) #seconds driver.get("http://www.xxxxx.com/loading") myDynamicElement = driver.find_element_by_id("myDynamicElement")
若是不设置,默认等待时间为0。