Python爬虫利器:Selenium的用法

发表于:2017-7-13 09:57

字体: | 上一篇 | 下一篇 | 我要投稿

 作者:未知    来源:博客

  简介
  Selenium 是什么?一句话,自动化测试工具。它支持各种浏览器,包括 Chrome,Safari,Firefox 等主流界面式浏览器,如果你在这些浏览器里面安装一个 Selenium 的插件,那么便可以方便地实现Web界面的测试。换句话说叫 Selenium 支持这些浏览器驱动。话说回来,PhantomJS不也是一个浏览器吗,那么 Selenium 支持不?答案是肯定的,这样二者便可以实现无缝对接了。
  然后又有什么好消息呢?Selenium支持多种语言开发,比如 Java,C,Ruby等等,有 Python 吗?那是必须的!哦这可真是天大的好消息啊。
  嗯,所以呢?安装一下 Python 的 Selenium 库,再安装好 PhantomJS,不就可以实现 Python+Selenium+PhantomJS 的无缝对接了嘛!PhantomJS 用来渲染解析JS,Selenium 用来驱动以及与 Python 的对接,Python 进行后期的处理,完美的三剑客!
  有人问,为什么不直接用浏览器而用一个没界面的 PhantomJS 呢?答案是:效率高!
  Selenium 有两个版本,目前最新版本是 2.53.1(2016/3/22)
  Selenium
  Selenium 2,又名 WebDriver,它的主要新功能是集成了 Selenium 1.0 以及 WebDriver(WebDriver 曾经是 Selenium 的竞争对手)。也就是说 Selenium 2 是 Selenium 和 WebDriver 两个项目的合并,即 Selenium 2 兼容 Selenium,它既支持 Selenium API 也支持 WebDriver API。
  更多详情可以查看 Webdriver 的简介。
   查看详情说明 Webdriver
  嗯,通过以上描述,我们应该对 Selenium 有了大概对认识,接下来就让我们开始进入动态爬取的新世界吧。
  本文参考内容来自 Selenium官网 SeleniumPython文档
  安装
  首先安装 Selenium
 pip install selenium
   快速开始
  初步体验
  我们先来一个小例子感受一下 Selenium,这里我们用 Chrome 浏览器来测试,方便查看效果,到真正爬取的时候换回 PhantomJS 即可。
  from selenium import webdriver

  browser = webdriver.Chrome()
  browser.get('http://www.baidu.com/')
  from selenium import webdriver
   运行这段代码,会自动打开浏览器,然后访问百度
  如果程序执行错误,浏览器没有打开,那么应该是没有装 Chrome 浏览器或者 Chrome 驱动没有配置在环境变量里。下载驱动,然后将驱动文件路径配置在环境变量即可。
  浏览器驱动下载:https://sites.google.com/a/chromium.org/chromedriver/downloads
  比如我的是 Mac OS,就把下载好的文件放在 /usr/bin 目录下就可以了。
  模拟提交
  下面的代码实现了模拟提交提交搜索的功能,首先等页面加载完成,然后输入到搜索框文本,点击提交。
  from selenium import webdriver
  from selenium.webdriver.common.keys import Keys

  driver = webdriver.Chrome()
  driver.get("http://www.python.org")
  assert "Python" in driver.title
  elem = driver.find_element_by_name("q")
  elem.send_keys("pycon")
  elem.send_keys(Keys.RETURN)
  print driver.page_source
   同样是在 Chrome 里面测试,感受一下。
  实例说明
  其中 driver.get 方法会打开请求的URL,WebDriver 会等待页面完全加载完成之后才会返回,即程序会等待页面的所有内容加载完成,JS渲染完毕之后才继续往下执行。注意:如果这里用到了特别多的 Ajax 的话,程序可能不知道是否已经完全加载完毕。
  WebDriver 提供了许多寻找网页元素的方法,譬如 find_element_by_* 的方法。例如一个输入框可以通过 find_element_by_name 方法寻找 name 属性来确定。然后我们输入来文本然后模拟点击了回车,就像我们敲击键盘一样。我们可以利用 Keys 这个类来模拟键盘输入。
  最后最重要的一点,获取网页渲染后的源代码,输出 page_source 属性即可。
   这样,我们就可以做到网页的动态爬取了。
  测试用例
  有了以上特性,我们当然可以用来写测试样例了。
  import unittest
  from selenium import webdriver
  from selenium.webdriver.common.keys import Keys

  class PythonOrgSearch(unittest.TestCase):

  def setUp(self):
  self.driver = webdriver.Chrome()

  def test_search_in_python_org(self):
  driver = self.driver
  driver.get("http://www.python.org")
  self.assertIn("Python", driver.title)
  elem = driver.find_element_by_name("q")
  elem.send_keys("pycon")
  elem.send_keys(Keys.RETURN)
  assert "No results found." not in driver.page_source

  def tearDown(self):
  self.driver.close()

  if __name__ == "__main__":
  unittest.main()
 
   运行程序,同样的功能,我们将其封装为测试标准类的形式。
  测试用例
  测试用例是继承了 unittest.TestCase 类,继承这个类表明这是一个测试类。setUp方法是初始化的方法,这个方法会在每个测试类中自动调用。每一个测试方法命名都有规范,必须以 test 开头,会自动执行。最后的 tearDown 方法会在每一个测试方法结束之后调用。这相当于最后的析构方法。在这个方法里写的是 close 方法,你还可以写 quit 方法。不过 close 方法相当于关闭了这个 TAB 选项卡,然而 quit 是退出了整个浏览器。当你只开启了一个 TAB 选项卡的时候,关闭的时候也会将整个浏览器关闭。

  页面操作
  页面交互

  仅仅抓取页面没有多大卵用,我们真正要做的是做到和页面交互,比如点击,输入等等。那么前提就是要找到页面中的元素。WebDriver提供了各种方法来寻找元素。例如下面有一个表单输入框。

<input type="text" name="passwd" id="passwd-id" />
   我们可以这样获取它
 element = driver.find_element_by_id("passwd-id")
  element = driver.find_element_by_name("passwd")
  element = driver.find_elements_by_tag_name("input")
  element = driver.find_element_by_xpath("//input[@id='passwd-id']")
   你还可以通过它的文本链接来获取,但是要小心,文本必须完全匹配才可以,所以这并不是一个很好的匹配方式。
  而且你在用 xpath 的时候还需要注意的是,如果有多个元素匹配了 xpath,它只会返回第一个匹配的元素。如果没有找到,那么会抛出 NoSuchElementException 的异常。
  获取了元素之后,下一步当然就是向文本输入内容了,可以利用下面的方法
  element.send_keys("some text")
  同样你还可以利用 Keys 这个类来模拟点击某个按键。
  element.send_keys("and some", Keys.ARROW_DOWN)
  你可以对任何获取到到元素使用 send_keys 方法,就像你在 GMail 里面点击发送键一样。不过这样会导致的结果就是输入的文本不会自动清除。所以输入的文本都会在原来的基础上继续输入。你可以用下面的方法来清除输入文本的内容。
  element.clear()
  这样输入的文本会被清除。
  填充表单
  我们已经知道了怎样向文本框中输入文字,但是其它的表单元素呢?例如下拉选项卡的的处理可以如下
   element = driver.find_element_by_xpath("//select[@name='name']")
  all_options = element.find_elements_by_tag_name("option")
  for option in all_options:
  print("Value is: %s" % option.get_attribute("value"))
  option.click()
  首先获取了第一个 select 元素,也就是下拉选项卡。然后轮流设置了 select 选项卡中的每一个 option 选项。你可以看到,这并不是一个非常有效的方法。
  其实 WebDriver 中提供了一个叫 Select 的方法,可以帮助我们完成这些事情。
   from selenium.webdriver.support.ui import Select
  select = Select(driver.find_element_by_name('name'))
  select.select_by_index(index)
  select.select_by_visible_text("text")
  select.select_by_value(value)
  如你所见,它可以根据索引来选择,可以根据值来选择,可以根据文字来选择。是十分方便的。
  全部取消选择怎么办呢?很简单
  select = Select(driver.find_element_by_id('id'))
  select.deselect_all()
  这样便可以取消所有的选择。
  另外我们还可以通过下面的方法获取所有的已选选项。
  select = Select(driver.find_element_by_xpath("xpath"))
  all_selected_options = select.all_selected_options
  获取所有可选选项是
  options = select.options
  如果你把表单都填好了,最后肯定要提交表单对吧。怎吗提交呢?很简单
  driver.find_element_by_id("submit").click()
  这样就相当于模拟点击了 submit 按钮,做到表单提交。
  当然你也可以单独提交某个元素
  element.submit()
  方法,WebDriver 会在表单中寻找它所在的表单,如果发现这个元素并没有被表单所包围,那么程序会抛出 NoSuchElementException 的异常。

21/212>
《2023软件测试行业现状调查报告》独家发布~

关注51Testing

联系我们

快捷面板 站点地图 联系我们 广告服务 关于我们 站长统计 发展历程

法律顾问:上海兰迪律师事务所 项棋律师
版权所有 上海博为峰软件技术股份有限公司 Copyright©51testing.com 2003-2024
投诉及意见反馈:webmaster@51testing.com; 业务联系:service@51testing.com 021-64471599-8017

沪ICP备05003035号

沪公网安备 31010102002173号