美文网首页菜鸟PythonPython开发
python爬虫的最佳实践(五)--selenium+Phant

python爬虫的最佳实践(五)--selenium+Phant

作者: Darkeril | 来源:发表于2016-05-05 19:51 被阅读52697次

Ps:又到了我们的ps环节,不知道上次大家尝试的如何,这次我们将简单介绍如何使用selenium+PhantomJS来抓取异步加载的网页数据信息。当然,selenium是一个非常强大的自动化工具,可以做非常多的事,有兴趣的同学可以自行了解一下。

这次我们的顺序稍稍变化一下,因为牵扯到配置环境。

环境配置

  • selenium
    pip install -U selenium
    建议采用pycharm安装,别忘了我们这个强大的IDE
  • PhantomJS
    这个不同的操作系统有各自对应的版本,这是官网的下载页面http://phantomjs.org/download.html,去下载你对应操作系统版本的phantomjs.下载完后,解压缩可以看到在文件夹的bin目录下有对应的phantomjs的可执行文件,拷贝一份放入一个环境变量可以搜索到的地方,或者直接把phantomjs的bin目录加入环境变量即可~

简单介绍一下PhantomJS,这是一个基于webkit的没有界面的浏览器,也就是它可以像浏览器解析网页,功能非常强大。但是据我测试。。解析的结果不一定和火狐或者chrome完全一样,但是完全够我们用。
简单介绍一下selenium,这是一个web的自动测试工具,可以模拟人的操作。支持市面上几乎所有的主流浏览器,同时也支持PhantomJS这种无界面浏览器。

因为selenium+Firefox或者Chrome太慢了,所以我们选用selenium+PhantomJS。
想试试selenium+Firefox的同学直接下载最新版的Firefox即可,不需要插件,但是chrome的话需要一个插件叫chromedriver,我会放到群里,闲话不多说,gogogo!

代码预览

#coding:utf-8
import unittest
from selenium import webdriver
from bs4 import BeautifulSoup


class seleniumTest(unittest.TestCase):
    def setUp(self):
        self.driver = webdriver.PhantomJS()

    def testEle(self):
        driver = self.driver
        driver.get('http://www.douyu.com/directory/all')
        soup = BeautifulSoup(driver.page_source, 'xml')
        while True:
            titles = soup.find_all('h3', {'class': 'ellipsis'})
            nums = soup.find_all('span', {'class': 'dy-num fr'})
            for title, num in zip(titles, nums):
                print title.get_text(), num.get_text()
            if driver.page_source.find('shark-pager-disable-next') != -1:
                break
            elem = driver.find_element_by_class_name('shark-pager-next')
            elem.click()
            soup = BeautifulSoup(driver.page_source, 'xml')

    def tearDown(self):
        print 'down'

if __name__ == "__main__":
    unittest.main()

小伙伴们是不是惊奇的发现今天的代码怎么略不一样了~因为今天除了爬虫相关,顺道教大家一些python 的小知识~

代码剖析

import unittest
什么是unittest呢?看字面意思估计大部分人已经懂了,就是单元测试的意思,这个库是python自带的给我们提供测试用的库,很好用,今天我顺道介绍一下这个库的简单使用。
今天的这段代码是一个简单的测试用例
class seleniumTest(unittest.TestCase):
继承TestCase类,表明这是一个测试类。setup方法是初始化方法,在unittest的main方法调用之初执行,然后tearDown是在测试完成的时候执行,中间的每个方法必须以test开头。这是一些简单的规则,大概了解之后我们就可以开心的使用单元测试了~

def setUp(self):
      self.dirver = webdriver.PhantomJS()

首先在我们的setup方法中初始化我们的webdriver,如果你用的是pycharm就可以看到webdriver后面会有多种方法,可以创建火狐的,chrome等各种浏览器的webdriver。

初始化完成之后开始解析,今天我们抓取斗鱼tv的房间信息,用浏览器打开http://www.douyu.com/directory/all,然后选择查看网页源代码,发现竟然可以直接在源码中看到房间信息,心中一喜,难道不是异步的方式么?这太好抓了,直接requests请求然后分析就可以了啊。紧接着再点开第二页,查看网页源代码。发现竟然和第一页一模一样,顿时失望,依然是采用的js异步的方式来加载数据的,只不过这次返回的不是json格式而直接是html。难道我们需要像上一章那样么?nono,今天我们直接使用PhantomJS模拟浏览器的访问过程,然后通过dirver.get方法获取浏览器加载完成后的源码。这样子无论是否异步,我们取到的source都是和我们在浏览器中看到的完全一样。是不是感觉简便了很多呢?

dirver.get('http://www.douyu.com/directory/all')
soup = BeautifulSoup(dirver.page_source, 'xml')

这两句想必大家也能理解了,其实就是换了一种方式去获取到网页源码而已。这次连带js渲染出来的网页也都获取到了。接下来

 titles = soup.find_all('h3', {'class': 'ellipsis'})
 nums = soup.find_all('span', {'class': 'dy-num fr'})
 for title, num in zip(titles, nums):
      print title.get_text(), num.get_text()

这段代码是在源码中取出我们想要的信息,可能同学们也发现了,这和我们第三节讲的不太一样,这是因为我们的解析方式换成了xml而不是原本的lxml了,lxml属于一种更加高效的解析模式,但是我为什么要更换呢。因为在我写这段代码的时候发生了一个小插曲,原本我也是用lxml的方式去解析的,但是发现每次只能解析出来32条信息,但是很明显斗鱼每个小分页里最少都有100多个房间,我查看了driver返回的page_source,发现没有问题,于是我想可能是beautifulsoup的解析出了问题,打印soup,发现只有源代码的一半,后面一半莫名奇妙丢失了。原本我以为是网页中有多个</html>标签导致解析出错,后来发现也不是这个问题,百度google很久也没有结果,所以只能换一种解析模式,当我换成xml的时候发现解析无误。暂时我也不知道原因,有兴趣的同学可以尝试解决一下这个问题。

那么上面这段代码取出了我们需要的信息,beautifulsoup的find方法要是不了解的话可以参看官方文档。接下来:

elem = driver.find_element_by_class_name('shark-pager-next') 
elem.click() 
soup = BeautifulSoup(driver.page_source, 'xml')

第一行代码是webdriver里面带有的定位标签的方法,我们通过观察发现斗鱼页面中下一页这个标签只出现一次,并且是独立样式,所以我们直接通过class进行定位,取到这个控件,然后执行element的click()方法模拟鼠标点击,这样,页面就自动跳到了下一页,接着继续解析网页源码。

if driver.page_source.find('shark-pager-disable-next') != -1: 
      break

这段代码是跳出循环用的,什么时候跳出循环呢?我们通过观察发现,当到最后一页时候斗鱼下一页的标签就会变灰,如图所示:

STU5R92IEHAZ{881KZ(72LD.png

于是,我们发现了有个唯一的shark-pager-disable-next样式,那么我们只需要在源码中找这个样式,如果有则说明已经解析到最后一页,直接跳出循环结束程序就好~

至此,今天的代码就讲解完毕了。有没有觉得比上次的观察法方便了很多呢?附带一提,有兴趣的通许可以吧今天代码中self.dirver = webdriver.PhantomJS()换成self.dirver = webdriver.Firefox(),看看会发生什么有趣的事情,我就不剧透了~

写在最后

惯例了,demo已经给你们了,也讲解完毕,现在需要你们自己实践了,去尝试抓取一下其他TV吧,推荐抓一下战旗,因为加载方式又和斗鱼完全不一样了,有挑战性点~下一节我们讲解如何在爬虫中使用多进程!

附加部分

一些seleium的小互动操作

  • 填入表单数据
#coding:utf-8
from selenium import webdriver
from selenium.webdriver.common.keys import Keys

driver = webdriver.Firefox()
driver.get('https://www.baidu.com/')
elem = driver.find_element_by_id('kw')
elem.send_keys(u'爬虫')
elem.send_keys(Keys.RETURN)
print(driver.page_source)
  • 滚动致页面最下方
#coding:utf-8
from selenium import webdriver
import time

driver = webdriver.Firefox()
driver.get('http://www.jianshu.com/collections')
time.sleep(1)
for i in range(10):
    dirver.execute_script('window.scrollTo(0,document.body.scrollHeight)')
    time.sleep(1)

使用selenium模拟人的操作行为,这次换成了火狐浏览器,可以看得更加清楚~

有兴趣的同学可以加群498945822一起交流学习哦~~
发现问题的同学欢迎指正,直接说就行,不用留面子,博主脸皮厚

相关文章

网友评论

  • 789f88991b67:我用phantomjs爬取js渲染的翻页,但是只能click一次,再次click就不行了
  • j4fan:楼主好,用phantomjs抓js页面遇到乱码怎么办呢?求回复,谢谢
  • 小幸运Q:弱弱的问一句......如果想给phathomjs动态切换代理的话该怎么办啊?(•̩̩̩̩_•̩̩̩̩)
  • jiang:很有帮助,谢谢!
  • 168954a01fbb:用beautifulsoup 只能抓静态的网页吧?
  • d4b53bcc4081:我的代码是这样的
    driver = webdriver.PhantomJS(executable_path='D:\\phantomjs-2.1.1-windows\\bin\\phantomjs.exe')
    def search(keyword):
    url_keyword = urllib.parse.quote(keyword)
    url = "http://wzdig.pbc.gov.cn:8080/dig/ui/search.action?q=&quot; + url_keyword
    # url = "http://www.tianyancha.com/search/&quot; + url_keyword + "?checkFrom=searchBox"
    print(url)
    driver.get(url)
    soup = BeautifulSoup(driver.page_source, "lxml")
    print(soup)
    search("123")
    但是它运行出来的结果依然还是js界面,这个怎么弄,求大神指点
  • Garfield_Liang:楼主的QQ群现在都没有人在。。。
  • vansnowpea:主题写的是PhantomJS ,但截图用的是火狐浏览器吗?
  • c8cb05d0cde1:跟楼上有同样的情况出现...
  • 85751fc48454:出了点问题,一直返回:
    NoSuchElementException: Message: Error Message => 'Unable to find element with class name 'shark-pager-next''


    elem = driver.find_element_by_class_name("shark-pager-next")
    改成
    elem = driver.find_element_by_css_selector("#J-pager > a.shark-pager-next")
    之后就还是找不到。
    NoSuchElementException: Message: Error Message => 'Unable to find element with css selector '#J-pager > a.shark-pager-next'' :cold_sweat:
  • 谢小路:@Darkeril 发现个问题,获取的title 和 num 并不是对应的,原因是新秀直播的title 和 在线直播的title 属性值相同... :smiley: 写的很好...
    5f81c4f8cecd:那就不要用find_all方法啊,我用的是soup.select方法,可以是对应的
    谢小路:@Darkeril 嗯呢,好好挺好使的,之前我不太熟悉这个异步加载,总是构造url...
    Darkeril:@谢小路 只是写个例子,都是临时想起来抓个斗鱼,所以没做区分

本文标题:python爬虫的最佳实践(五)--selenium+Phant

本文链接:https://www.haomeiwen.com/subject/svzlrttx.html