python写爬虫3-MongoDB数据缓存(采集58出租房信息)

2023-12-07 03:20

本文主要是介绍python写爬虫3-MongoDB数据缓存(采集58出租房信息),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

python写爬虫3-MongoDB数据缓存(采集58出租房信息)

有时,我们下载某个页面并抓取部分数据后,之后可能还会需要重新下载该页面,抓取其他数据。对于小网站而言,这不算什么大问题;但对于那些拥有百万网页的网站来说,重新爬取可能需要耗费大量时间。因此,我们可以对已爬取的网页进行缓存,让每个页面只下载一次。

本文代码只是实现了数据的存储与获取,与上述案例实情不符
开发环境:
1.硬件mac
2.python2.7
3.MongoDB3.4.2

NoSQL即Not Only SQL,通常是无模式的,NoSQL包含列数据存储(HBase),键值对存储(Redis),面向文档的数据库(MongoDB),图形数据库(Neo4j);本文采用MongoDB,MongoDB有个功能:为数据设定时间,当到达设定时间后,MongoDB可自动为我们删除记录。

需注意,MongoDB缓存无法按设定时间精确清理过期记录,会存在1分钟内的延迟,这是MongoDB的运行机制造成的

1.安装MongoDB及对应的Python封装库

mac下安装MongoDB命令:

brew update
brew install mongodb
pip install pymongo

2.配置MongoDB

创建默认MongoDB数据库文件存放目录

mkdir -p /data/db

记得给此目录加用户的读写权限,不然会报权限错误;其他配置,请参见官方文档。

3.启动MongoDB

mongod

4.MongoDB缓存实现

#!/usr/bin/env python
# -*- coding: utf-8 -*-import pickle
import zlib
from bson.binary import Binary
from datetime import datetime, timedelta
from pymongo import MongoClientclass MongoCache:def __init__(self, client=None, expires=timedelta(days=1)):self.client = MongoClient('localhost', 27017)self.db = self.client.cacheself.db.webpage.create_index('timestamp', expireAfterSeconds=expires.total_seconds())def __getitem__(self, item):record = self.db.webpage.find_one({'id': item})if record:return pickle.loads(zlib.decompress(record['result']))  # 压缩数据else:return None# raise KeyError(item + 'dose not exist')def __setitem__(self, key, value):record = {'result': Binary(zlib.compress(pickle.dumps(value))), 'timestamp': datetime.utcnow()}self.db.webpage.update({'id': key}, {'$set': record}, upsert=True)

5.爬虫编写

先看一下http://bj.58.com/zufang/页面结构:
这里写图片描述
再看一下出租房详细页面结构:
这里写图片描述
下面开始敲代码:

#! /usr/bin/env python
# -*- coding:utf-8 -*-import urllib2
import lxml.html
import time
from lxml.cssselect import CSSSelector
from MongoCache import MongoCachedef download(url, user_agent='Google', num_retries=2):"""下载整个页面"""print 'Downloading:', url# 设置用户代理headers = {'User-agent': user_agent}request = urllib2.Request(url, headers=headers)try:html = urllib2.urlopen(request).read()except urllib2.URLError as e:print 'Downloading error:', e.reasonhtml = None# 只有在服务器报500-600错误时,才会重试下载,仅重试2次if num_retries > 0:if hasattr(e, 'code') and 500 <= e.code < 600:return download(url, num_retries-1)return htmldef get_data(url):"""从详细页面 获取各字段数据"""#  如果缓存中有该页面数据,则直接获取使用;否则,先下载页面,再使用cache = MongoCache()if not cache.__getitem__(url):html_text_detail = download(url)if not html_text_detail:passelse:cache.__setitem__(url, html_text_detail)else:print 'Exists:', urlhtml_text_detail = cache.__getitem__(url)try:#  获取个字段数据tree = lxml.html.fromstring(html_text_detail)house_title = CSSSelector('div.main-wrap > div.house-title > h1')house_pay_way1 = CSSSelector('div.house-pay-way > span:nth-child(1)')house_pay_way2 = CSSSelector('div.house-pay-way > span:nth-child(2)')print house_title(tree)[0].text_content()print '%s|%s' % (house_pay_way1(tree)[0].text_content(), house_pay_way2(tree)[0].text_content())for i in range(7):for j in range(2):css = 'div.house-desc-item > ul.f14 > li:nth-child(%s) > span:nth-child(%s)' % (i+1, j+1)house_info = CSSSelector(css)print house_info(tree)[0].text_content().replace(' ', '')except TypeError as e:print 'HTML文本发生错误:%s' % eexcept IndexError as e:print '获取详细数据发生错误:%s' % edef get_url(html):"""获取需爬取数据的链接集"""tree = lxml.html.fromstring(html)sel = CSSSelector('div.mainbox > div.main > div.content > div.listBox > ul.listUl > li > div.des > h2 > a')url_list = []for i in sel(tree):if i.get('href') not in url_list:url_list.append(i.get('href'))return url_listif __name__ == '__main__':url_index = 'http://bj.58.com/chuzu/'html_text_list = download(url_index)url_list = get_url(html_text_list)for url_detail in url_list:time.sleep(2)  # 延时2sget_data(url_detail)

执行效果图:(左屏为数据抓取输出;右屏为MongoDB数据库)
这里写图片描述

这篇关于python写爬虫3-MongoDB数据缓存(采集58出租房信息)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:https://blog.csdn.net/apple9005/article/details/54967916
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/464322

相关文章

Python中OpenCV与Matplotlib的图像操作入门指南

《Python中OpenCV与Matplotlib的图像操作入门指南》:本文主要介绍Python中OpenCV与Matplotlib的图像操作指南,本文通过实例代码给大家介绍的非常详细,对大家的学... 目录一、环境准备二、图像的基本操作1. 图像读取、显示与保存 使用OpenCV操作2. 像素级操作3.

Python使用FFmpeg实现高效音频格式转换工具

《Python使用FFmpeg实现高效音频格式转换工具》在数字音频处理领域,音频格式转换是一项基础但至关重要的功能,本文主要为大家介绍了Python如何使用FFmpeg实现强大功能的图形化音频转换工具... 目录概述功能详解软件效果展示主界面布局转换过程截图完成提示开发步骤详解1. 环境准备2. 项目功能结

解决mysql插入数据锁等待超时报错:Lock wait timeout exceeded;try restarting transaction

《解决mysql插入数据锁等待超时报错:Lockwaittimeoutexceeded;tryrestartingtransaction》:本文主要介绍解决mysql插入数据锁等待超时报... 目录报错信息解决办法1、数据库中执行如下sql2、再到 INNODB_TRX 事务表中查看总结报错信息Lock

使用Python实现Windows系统垃圾清理

《使用Python实现Windows系统垃圾清理》Windows自带的磁盘清理工具功能有限,无法深度清理各类垃圾文件,所以本文为大家介绍了如何使用Python+PyQt5开发一个Windows系统垃圾... 目录一、开发背景与工具概述1.1 为什么需要专业清理工具1.2 工具设计理念二、工具核心功能解析2.

使用C#删除Excel表格中的重复行数据的代码详解

《使用C#删除Excel表格中的重复行数据的代码详解》重复行是指在Excel表格中完全相同的多行数据,删除这些重复行至关重要,因为它们不仅会干扰数据分析,还可能导致错误的决策和结论,所以本文给大家介绍... 目录简介使用工具C# 删除Excel工作表中的重复行语法工作原理实现代码C# 删除指定Excel单元

Java实现本地缓存的常用方案介绍

《Java实现本地缓存的常用方案介绍》本地缓存的代表技术主要有HashMap,GuavaCache,Caffeine和Encahche,这篇文章主要来和大家聊聊java利用这些技术分别实现本地缓存的方... 目录本地缓存实现方式HashMapConcurrentHashMapGuava CacheCaffe

Linux lvm实例之如何创建一个专用于MySQL数据存储的LVM卷组

《Linuxlvm实例之如何创建一个专用于MySQL数据存储的LVM卷组》:本文主要介绍使用Linux创建一个专用于MySQL数据存储的LVM卷组的实例,具有很好的参考价值,希望对大家有所帮助,... 目录在Centos 7上创建卷China编程组并配置mysql数据目录1. 检查现有磁盘2. 创建物理卷3. 创

Python实现一键PDF转Word(附完整代码及详细步骤)

《Python实现一键PDF转Word(附完整代码及详细步骤)》pdf2docx是一个基于Python的第三方库,专门用于将PDF文件转换为可编辑的Word文档,下面我们就来看看如何通过pdf2doc... 目录引言:为什么需要PDF转Word一、pdf2docx介绍1. pdf2docx 是什么2. by

Python函数返回多个值的多种方法小结

《Python函数返回多个值的多种方法小结》在Python中,函数通常用于封装一段代码,使其可以重复调用,有时,我们希望一个函数能够返回多个值,Python提供了几种不同的方法来实现这一点,需要的朋友... 目录一、使用元组(Tuple):二、使用列表(list)三、使用字典(Dictionary)四、 使

Python程序的文件头部声明小结

《Python程序的文件头部声明小结》在Python文件的顶部声明编码通常是必须的,尤其是在处理非ASCII字符时,下面就来介绍一下两种头部文件声明,具有一定的参考价值,感兴趣的可以了解一下... 目录一、# coding=utf-8二、#!/usr/bin/env python三、运行Python程序四、