爬虫请求库——selenium

Posted ゛竹先森゜

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了爬虫请求库——selenium相关的知识,希望对你有一定的参考价值。

selenium模块

  selenium 最初是一个自动化测试工具,而爬虫中使用它主要是为了解决 requests 无法直接执行 javascript 代码的问题。selenium 的缺点是效率会变得很慢。

  selenium 本质是通过驱动浏览器,完全模拟浏览器的操作,比如跳转、输入、点击、下拉等,来拿到网页渲染之后的结果,可支持多种浏览器

from selenium import webdriver
browser=webdriver.Chrome()
browser=webdriver.Firefox()
browser=webdriver.PhantomJS()
browser=webdriver.Safari()
browser=webdriver.Edge()

一、安装

安装:

pip3 install selenium

要自动启动浏览器需要安装相应的驱动,selenium3 默认支持的 webdriver 是 Firfox,而 Firefox 需要安装 geckodriver

 

火狐浏览器安装 driver

下载链接:https://github.com/mozilla/geckodriver/releases

  • Linux 下载后将 geckodriver 移动到 /usr/local/bin/ 目录下(或者软连接到 /usr/bin/chromdriver)
  • Windows下载后将 geckodriver.exe  加入环境变量,或者将 geckodriver.exe  移动到脚本同级目录

谷歌浏览器安装 driver:

  • windows:下载 chromdriver.exe 放到python安装路径的 scripts 目录中即可,注意最新版本是2.xx,并非2.9
  • Linux:下载 chromdriver.zip,解压缩后将 chromdriver 文件复制到 /usr/bin 文件夹下(或者软连接到 /usr/bin/chromdriver)

  国内镜像网站地址:http://npm.taobao.org/mirrors/chromedriver/
  最新的版本去官网找:https://sites.google.com/a/chromium.org/chromedriver/downloads

from selenium import webdriver
# 弹出 chrome 浏览器
driver=webdriver.Chrome()
driver.get(\'https://www.baidu.com\')
driver.page_source
#能自动弹出浏览器并访问百度,就说明成功了

# 弹出 firefox浏览器
brow=webdriver.Firefox()
brow.get(\'https://www.baidu.com\') 

 

#安装:selenium+phantomjs
pip3 install selenium
下载phantomjs,解压后把phantomjs.exe所在的bin目录放到环境变量
下载链接:http://phantomjs.org/download.html

#验证安装
C:\\Users\\Administrator>phantomjs
phantomjs> console.log(\'egon gaga\')
egon gaga
undefined
phantomjs> ^C
C:\\Users\\Administrator>python3
Python 3.6.1 (v3.6.1:69c0db5, Mar 21 2017, 18:41:36) [MSC v.1900 64 bit (AMD64)] on win32
Type "help", "copyright", "credits" or "license" for more information.
>>> from selenium import webdriver
>>> driver=webdriver.PhantomJS() #无界面浏览器
>>> driver.get(\'https://www.baidu.com\')
>>> driver.page_source
无界面浏览器

 

二、等待元素被加载

  selenium 是自动运行浏览器的一种行为,而浏览器的加载是需要时间的,我们在进行操作之前肯定必须要等到被操作的元素加载出来后才行,所以我们就需要用到等待的操作、

等待的方式分两种

#2、等待的方式分两种:
隐式等待:在browser.get(\'xxx\')前就设置,针对所有元素有效
显式等待:在browser.get(\'xxx\')之后设置,只针对某个元素有效

示例:

from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素

browser=webdriver.Chrome()#启动浏览器
"""
方式一
隐式等待:在查找所有元素时,如果尚未被加载,则等10秒,在最前面设置
"""
browser.implicitly_wait(10)

browser.get(\'https://www.baidu.com\')


input_tag=browser.find_element_by_id(\'kw\')
input_tag.send_keys(\'美女\')
input_tag.send_keys(Keys.ENTER)



"""
方式二
显式等待:显式地等待某个指定元素被加载
这两条效果与上面一条相同
"""
#wait=WebDriverWait(browser,10)
#wait.until(EC.presence_of_element_located((By.ID,\'content_left\')))


#查找内容
contents=browser.find_element_by_id(\'content_left\') #没有进行等待操作就直接查找的话,找不到就会报错
print(contents)

browser.close()
View Code

 

三、选择器

  选择器就是用来做对象爬取的内容进行定位用的,选择器有很多,每种的功能都不同

基本选择器

#官网链接:http://selenium-python.readthedocs.io/locating-elements.html
from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素
import time

driver=webdriver.Chrome()
driver.get(\'https://www.baidu.com\')
wait=WebDriverWait(driver,10)

try:
    #===============所有方法===================
    # 1、find_element_by_id
    # 2、find_element_by_link_text
    # 3、find_element_by_partial_link_text  #模糊匹配
    # 4、find_element_by_tag_name
    # 5、find_element_by_class_name
    # 6、find_element_by_name
    # 7、find_element_by_css_selector
    # 8、find_element_by_xpath  #后续详解
    # 强调:
    # 1、上述均可以改写成find_element(By.ID,\'kw\')的形式
    # 2、find_elements_by_xxx的形式是查找到多个元素,结果为列表

    #===============示范用法===================
    # 1、find_element_by_id
    print(driver.find_element_by_id(\'kw\'))

    # 2、find_element_by_link_text
    # login=driver.find_element_by_link_text(\'登录\')
    # login.click()

    # 3、find_element_by_partial_link_text
    login=driver.find_elements_by_partial_link_text(\'\')[0]
    login.click()

    # 4、find_element_by_tag_name
    print(driver.find_element_by_tag_name(\'a\'))

    # 5、find_element_by_class_name
    button=wait.until(EC.element_to_be_clickable((By.CLASS_NAME,\'tang-pass-footerBarULogin\')))
    button.click()

    # 6、find_element_by_name
    input_user=wait.until(EC.presence_of_element_located((By.NAME,\'userName\')))
    input_pwd=wait.until(EC.presence_of_element_located((By.NAME,\'password\')))
    commit=wait.until(EC.element_to_be_clickable((By.ID,\'TANGRAM__PSP_10__submit\')))

    input_user.send_keys(\'18611453110\')
    input_pwd.send_keys(\'lhf@094573\')
    commit.click()

    # 7、find_element_by_css_selector
    driver.find_element_by_css_selector(\'#kw\')

    # 8、find_element_by_xpath

    time.sleep(5)

finally:
    driver.close()
基本选择器

xpath

#官网链接:http://selenium-python.readthedocs.io/locating-elements.html
from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素
import time

driver=webdriver.PhantomJS()
driver.get(\'https://doc.scrapy.org/en/latest/_static/selectors-sample1.html\')
# wait=WebDriverWait(driver,3)
driver.implicitly_wait(3) #使用隐式等待

try:
    # find_element_by_xpath
    #//与/的不同
    # driver.find_element_by_xpath(\'//body/a\')  # 开头的//代表从整篇文档中寻找,body之后的/代表body的儿子,这一行找不到就会报错了

    driver.find_element_by_xpath(\'//body//a\')  # 开头的//代表从整篇文档中寻找,body之后的//代表body的子子孙孙
    driver.find_element_by_css_selector(\'body a\')

    #取第n个
    res1=driver.find_elements_by_xpath(\'//body//a[1]\') #取第一个a标签,这里的索引是从1开始
    print(res1[0].text)

    #按照属性查找,下述三者查找效果一样
    res1=driver.find_element_by_xpath(\'//a[5]\')
    res2=driver.find_element_by_xpath(\'//a[@href="image5.html"]\')
    res3=driver.find_element_by_xpath(\'//a[contains(@href,"image5")]\') #模糊查找
    print(\'==>\', res1.text)
    print(\'==>\',res2.text)
    print(\'==>\',res3.text)


    #其他
    res1=driver.find_element_by_xpath(\'/html/body/div/a\')
    print(res1.text)

    res2=driver.find_element_by_xpath(\'//a[img/@src="image3_thumb.jpg"]\') #找到子标签img的src属性为image3_thumb.jpg的a标签
    print(res2.tag_name,res2.text)

    res3 = driver.find_element_by_xpath("//input[@name=\'continue\'][@type=\'button\']") #查看属性name为continue且属性type为button的input标签
    res4 = driver.find_element_by_xpath("//*[@name=\'continue\'][@type=\'button\']") #查看属性name为continue且属性type为button的所有标签
    
    
    time.sleep(5)

finally:
    driver.close()
xpath

获取标签属性

from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素

browser=webdriver.Chrome()

browser.get(\'https://www.amazon.cn/\')

wait=WebDriverWait(browser,10)
wait.until(EC.presence_of_element_located((By.ID,\'cc-lm-tcgShowImgContainer\')))

tag=browser.find_element(By.CSS_SELECTOR,\'#cc-lm-tcgShowImgContainer img\')

#获取标签属性,
print(tag.get_attribute(\'src\'))


#获取标签ID,位置,名称,大小(了解)
print(tag.id)
print(tag.location)
print(tag.tag_name)
print(tag.size)


browser.close()
获取标签的属性

 

四、元素交互操作

  交互操作就是指模拟人在网页中进行输入或点击鼠标的操作,主要针对的是 input 框和链接。交互操作中可以自己植入 js 代码

模拟天猫搜索:

from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素
import time

driver=webdriver.Chrome()
driver.get(\'https://www.tmall.com/\')
driver.implicitly_wait(3)

try:
    input_tag=driver.find_element_by_id(\'mq\')#搜索框
    input_tag.send_keys(\'NIKE\')#写入搜索条件
    input_tag.send_keys(Keys.ENTER)#回车

    time.sleep(3)
    input_tag=driver.find_element_by_id(\'mq\')
    input_tag.clear()#清空搜索框
    input_tag.send_keys(\'科比战靴\')
    input_tag.send_keys(Keys.ENTER)


    time.sleep(5)
finally:
    driver.close()
View Code

模拟滑动验证码操作(网页很low逼):

from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素
import time

driver=webdriver.Chrome()
driver.get(\'http://www.runoob.com/try/try.php?filename=jqueryui-api-droppable\')
#driver.get(\'http://www.baidu.com\')
driver.implicitly_wait(3)

try:
    driver.switch_to.frame(\'iframeResult\')#已被弃用的html标签,作用是html下套html
    driver.switch_to.parent_frame()#返回父级html标签

    source=driver.find_element_by_id(\'draggable\')#源滑块
    target=driver.find_element_by_id(\'droppable\')#目标
    print(source,target)

    #方式一:
    # actions=ActionChains(driver)
    # actions.drag_and_drop(source,target)
    # actions.perform()

    #方式二:
    distance=target.location[\'x\']-source.location[\'x\']#距离
    ActionChains(driver).click_and_hold(source).perform()#模拟按住鼠标不放手

    print(distance)
    s=0
    while s < distance:
        print(s)
        ActionChains(driver).move_by_offset(xoffset=2,yoffset=0).perform()#每次移动2像素
        s+=2
    ActionChains(driver).release().perform()#释放鼠标


    driver.execute_script(\'alert("xxxxxxxxx")\')#可自写js代码,会执行
    time.sleep(6)
finally:
    driver.close()
View Code

补充:frame的切换

#frame相当于一个单独的网页,在父frame里是无法直接查看到子frame的元素的,必须switch_to_frame切到该frame下,才能进一步查找

from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By #按照什么方式查找,By.ID,By.CSS_SELECTOR
from selenium.webdriver.common.keys import Keys #键盘按键操作
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait #等待页面加载某些元素


try:
    browser=webdriver.Chrome()
    browser.get(\'http://www.runoob.com/try/try.php?filename=jqueryui-api-droppable\')

    browser.switch_to.frame(\'iframeResult\') #切换到id为iframeResult的frame

    tag1=browser.find_element_by_id(\'droppable\')
    print(tag1)

    # tag2=browser.find_element_by_id(\'textareaCode\') #报错,在子frame里无法查看到父frame的元素
    browser.switch_to.parent_frame() #切回父frame,就可以查找到了
    tag2=browser.find_element_by_id(\'textareaCode\')
    print(tag2)

finally:
    browser.close()
补充:frame的切换

模拟浏览器的前进后退

#模拟浏览器的前进后退
import time
from selenium import webdriver

browser=webdriver.Chrome()
browser.get(\'https://www.baidu.com\')
browser.get(\'https://www.taobao.com\')
browser.get(\'http://www.sina.com.cn/\')

browser.back()#后腿
time.sleep(10)
browser.forward()#前进
browser.close()
模拟浏览器的前进后退
#cookies
from selenium import webdriver

browser=webdriver.Chrome()
browser.get(\'https://www.zhihu.com/explore\')
print(browser.get_cookies())
browser.add_cookie({\'k1\':\'xxx\',\'k2\':\'yyy\'})
print(browser.get_cookies())

# browser.delete_all_cookies()
设置和得到cookies

选项卡管理

#选项卡管理:切换选项卡,有js的方式windows.open,有windows快捷键:ctrl+t等,最通用的就是js的方式
import time
from selenium import webdriver

browser=webdriver.Chrome()
browser.get(\'https://www.baidu.com\')
browser.execute_script(\'window.open()\')

print(browser.window_handles) #获取所有的选项卡
browser.switch_to_window(browser.window_handles[1])
browser.get(\'https://www.taobao.com\')
time.sleep(10)
browser.switch_to_window(browser.window_handles[0])
browser.get(\'https://www.sina.com.cn\')
browser.close()
选项卡管理

异常处理

from selenium import webdriver
from selenium.common.exceptions import TimeoutException,NoSuchElementException,NoSuchFrameException

try:
    browser=webdriver.Chrome()
    browser.get(\'http://www.runoob.com/try/try.php?filename=jqueryui-api-droppable\')
    browser.switch_to.frame(\'iframssseResult\')

except TimeoutException as e:
    print(e)
except NoSuchFrameException as e:
    print(e)
finally:
    browser.close()
异常处理

 

小项目扩展

#注意:网站都策略都是在不断变化的,精髓在于学习流程。下述代码生效与2017-11-7,不能保证永久有效
from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait

browser=webdriver.Chrome()

try:
    browser.get(\'http://mail.163.com/\')

    wait=WebDriverWait(browser,5)

    frame=wait.until(EC.presence_of_element_located((By.ID,\'x-URS-iframe\')))
    browser.switch_to.frame(frame)

    wait.until(EC.presence_of_element_located((By.CSS_SELECTOR,\'.m-container\')))

    inp_user=browser.find_element_by_name(\'email\')
    inp_pwd=browser.find_element_by_name(\'password\')
    button=browser.find_element_by_id(\'dologin\')
    inp_user.send_keys(\'18611453110\')
    inp_pwd.send_keys(\'xxxx\')
    button.click()

    #如果遇到验证码,可以把下面一小段打开注释
    # import time
    # time.sleep(10)
    # button = browser.find_element_by_id(\'dologin\')
    # button.click()

    wait.until(EC.presence_of_element_located((By.ID,\'dvNavTop\')))
    write_msg=browser.find_elements_by_css_selector(\'#dvNavTop li\')[1] #获取第二个li标签就是“写信”了
    write_msg.click()


    wait.until(EC.presence_of_element_located((By.CLASS_NAME,\'tH0\')))
    recv_man=browser.find_element_by_class_name(\'nui-editableAddr-ipt\')
    title=browser.find_element_by_css_selector(\'.dG0 .nui-ipt-input\')
    recv_man.send_keys(\'378533872@qq.com\')
    title.send_keys(\'圣旨\')
    print(title.tag_name)


    frame=wait.until(EC.presence_of_element_located((By.CLASS_NAME,\'APP-editor-iframe\')))
    browser.switch_to.frame(frame)
    body=browser.find_element(By.CSS_SELECTOR,\'body\')
    body.send_keys(\'egon很帅,可以加工资了\')

    browser.switch_to.parent_frame() #切回他爹
    send_button=browser.find_element_by_class_name(\'爬虫请求库——requests

请求库之 selenium模块

爬虫系列

selenium模块的基本使用

python爬虫需要安装哪些库

requests库结合selenium库共同完成web自动化和爬虫工作