JS动态加载以及JavaScript void的爬虫解决方案

Posted 程序员的碎碎念

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JS动态加载以及JavaScript void的爬虫解决方案相关的知识,希望对你有一定的参考价值。

# Intro

本文以 Chrome浏览器为工具, 36Kr为示例网站, 使用 Json Handle 作为辅助信息解析工具, 演示如何抓取此类网站.


# Detail

Step 1. 按下 F12 或右键 检查进入开发者工具


Step 2. 选中Network一栏, 筛选 XHR请求

XHRXMLHttpRequest, 可以异步或同步返回服务器响应的请求, 并且能够以文本或者一个 DOM 文档的形式返回内容.


JSON是一种与XML在格式上很像, 但是占用空间更小的数据交换格式, 全程是 javascript Object Notation, 本文中的36Kr动态加载时获取到的信息就是JSON类型的数据.

网站为了节省空间, 加快响应, 常常没有对 JSON 进行格式化, 导致 JSON 的可读性差, 难以寻找我们要的信息.


我们通过右键打开获取到的 XHR 请求, 然后看看数据是怎样的

JS动态加载以及JavaScript void(0)的爬虫解决方案

JS动态加载以及JavaScript void(0)的爬虫解决方案

JS动态加载以及JavaScript void(0)的爬虫解决方案

使用 Json Handle 后的数据可读性就很高了


Step 3. 分析 URL

结合上面的截图, 分析这条 URL

https://36kr.com/api/newsflash?column_ids=69&no_bid=false&b_id=126035&per_page=20&_=1530699384159


这中间有两个参数很容易可以知道它的用途, 第一个是 per_page=20, 第二个是 _=1530699384159第一个参数是我们每次滚动后可以获取到的信息条数, 第二个是时间戳


试着改第一个参数改为10, 可以看到条数就变为10了. JS动态加载以及JavaScript void(0)的爬虫解决方案

改为1000呢? 很遗憾, 最大值只有300. 换算下来, 就是最多允许爬 15 页


滑动了超过15页发现仍然有信息显示, 经过转换, 发现它的时间戳只是浏览网页生成的时间戳, 与内容无关

按了几个数字, 修改了 b_id的值, 发现内容确实发生了改变, b_id又是网站设定的规则, 无从入手

JS动态加载以及JavaScript void(0)的爬虫解决方案


改了 no_bidtrue似乎没有变化, 接着修改了 column_id为70, 发现新闻的内容发生改变, 合理猜测这个应该是新闻标签的id. JS动态加载以及JavaScript void(0)的爬虫解决方案


至此, 我们已大致了解整个 URL 的含义

per_page 每次滑动可以获得的数据条目, 最大值为300

column_ids 新闻内容标签, 69为资本, 68为B轮后等

b_id 新闻集合的某种id

时间戳 记录当前的浏览时间

最后把原本的 URL 缩减为 

https://36kr.com/api/newsflash?column_ids=69&no_bid=true&b_id=&per_page=300


舍弃了 b_id, 同时删去时间戳, 防止服务器发现每次接收到的请求时间都是一样的


经过测试, 上述的 URL 是可以获取信息的


Step 4. 开始爬虫

接下来的步骤与平时爬虫类似. 不同的是获取信息不再通过Xpath这些工具, 而是直接通过 JSON 取值

取值方式简单粗暴, 点击对应的内容就可以看路径了


接着用 scrapy shell工具测试下正确性, 然后就可以写代码了.

由于新闻来源隐藏在 description, 经过观察, 不难发现它的规律, 写一条正则获取即可, 如果结果为空, 则说明来源是36Kr 


# Source Code

Spider

 
   
   
 
  1. # -*- coding: utf-8 -*-

  2. import scrapy

  3. import json

  4. import re

  5. from scrapy import Request

  6. from ..items import FinvestItem

  7. class A36krSpider(scrapy.Spider):

  8.    name = '36kr'

  9.    allowed_domains = ['36kr.com']

  10.    start_urls = ['https://36kr.com/api/newsflash?column_ids=69&no_bid=true&b_id=&per_page=300']

  11.    headers = {

  12.        'User-Agent': 'Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (Khtml, like Gecko) Chrome/53.0.2785.143 Safari/537.36',

  13.    }

  14.    def start_request(self):

  15.        yield Request(self.start_urls, headers=self.headers)

  16.    def parse(self, response):

  17.        item = FinvestItem()

  18.        # 转化为 unicode 编码的数据

  19.        sites = json.loads(response.body_as_unicode())

  20.        src_pattern = re.compile('。((.*))')

  21.        for i in sites['data']['items']:

  22.            item['link'] = i['news_url']

  23.            item['title'] = i['title']

  24.            if src_pattern.search(i['description']) == None:

  25.                item['source'] = "36Kr"

  26.            else:

  27.                item['source'] = src_pattern.search(i['description']).group(1)

  28.            item['create_time'] = i['published_at']

  29.            item['content'] = i['description']

  30.            yield item

Pipeline

 
   
   
 
  1. # -*- coding: utf-8 -*-

  2. # Define your item pipelines here

  3. #

  4. # Don't forget to add your pipeline to the ITEM_PIPELINES setting

  5. # See: https://doc.scrapy.org/en/latest/topics/item-pipeline.html

  6. import pymongo

  7. import re

  8. from scrapy.conf import settings

  9. class FinvestPipeline(object):

  10.    def __init__(self):

  11.        """

  12.        use for connecting to mongodb

  13.        """

  14.        # connect to db

  15.        self.client = pymongo.MongoClient(host=settings['MONGO_HOST'], port=settings['MONGO_PORT'])

  16.        # ADD if NEED account and password

  17.        # self.client.admin.authenticate(host=settings['MONGO_USER'], settings['MONGO_PSW'])

  18.        self.db = self.client[settings['MONGO_DB']]

  19.        self.coll = self.db[settings['MONGO_COLL']]

  20.    def process_item(self, item, spider):

  21.        content = item['content']

  22.        title = item['title']

  23.        fin = re.compile(r'(?:p|P)re-?(?:A|B)轮|(?:A|B|C|D|E)+?1?2?3?轮|(?:天使轮|种子|首)轮|IPO|轮|(?:p|Pre)IPO')

  24.        result = fin.findall(title)

  25.        if(len(result) == 0):

  26.            result = "未透露"

  27.        else:

  28.            result = ''.join(result)

  29.        content = content.replace(u'<p>', u' ').replace(u'</p>', u' ').replace(u' ', ' ').strip()

  30.        # delete html label in content

  31.        rule = re.compile(r'<[^>]+>', re.S)

  32.        content = rule.sub('', content)

  33.        item['content'] = content

  34.        item['funding_round'] = result

  35.        self.coll.insert(dict(item))

  36.        return item

抓取网站:https://36kr.com/newsflashes


入门小白, 欢迎大家指出错误, 技术交流

https://github.com/FesonX/finvest-spider



以上是关于JS动态加载以及JavaScript void的爬虫解决方案的主要内容,如果未能解决你的问题,请参考以下文章

以及ajax以及angularjs 动态模板加载并进行渲染

“高三”笔记之动态JS动态样式

JavaScript 之 动态加载JS代码或JS文件

python+selenium+PhantomJS爬取网页动态加载内容

JavaScript_动态加载CSS和JS文件

javascript [js:重写地址栏]动态重写地址栏,无需重新加载。 #js