浅学爬虫-爬虫维护与优化

在实际项目中,爬虫的稳定性和效率至关重要。通过错误处理与重试机制、定时任务以及性能优化,可以确保爬虫的高效稳定运行。下面我们详细介绍这些方面的技巧和方法。

错误处理与重试机制

在爬虫运行过程中,网络不稳定、目标网站变化等因素可能会导致请求失败。为了确保爬虫的健壮性,需要实现错误处理与重试机制。

示例:实现错误处理与重试机制

我们将修改之前的新闻爬虫示例,加入错误处理与重试机制。

python 复制代码
import requests
from bs4 import BeautifulSoup
import csv
import time

# 文章列表页URL模板
base_url = "http://news.example.com/page/"
max_retries = 3  # 最大重试次数

# 爬取文章详情的函数
def fetch_article(url):
    for attempt in range(max_retries):
        try:
            response = requests.get(url)
            response.raise_for_status()
            soup = BeautifulSoup(response.content, 'html.parser')
            title = soup.find('h1', class_='article-title').text
            author = soup.find('span', class_='article-author').text
            date = soup.find('span', class_='article-date').text
            content = soup.find('div', class_='article-content').text
            return {
                'title': title,
                'author': author,
                'date': date,
                'content': content
            }
        except requests.exceptions.RequestException as e:
            print(f"请求失败: {e},重试 {attempt + 1} 次...")
            time.sleep(2 ** attempt)  # 指数退避算法
    return None

# 爬取文章列表页的函数
def fetch_articles_from_page(page):
    url = f"{base_url}{page}"
    for attempt in range(max_retries):
        try:
            response = requests.get(url)
            response.raise_for_status()
            articles = []
            soup = BeautifulSoup(response.content, 'html.parser')
            links = soup.find_all('a', class_='article-link')
            for link in links:
                article_url = link['href']
                article = fetch_article(article_url)
                if article:
                    articles.append(article)
            return articles
        except requests.exceptions.RequestException as e:
            print(f"请求失败: {e},重试 {attempt + 1} 次...")
            time.sleep(2 ** attempt)  # 指数退避算法
    return []

# 保存数据到CSV文件
def save_to_csv(articles, filename):
    with open(filename, 'w', newline='', encoding='utf-8') as csvfile:
        fieldnames = ['title', 'author', 'date', 'content']
        writer = csv.DictWriter(csvfile, fieldnames=fieldnames)
        writer.writeheader()
        for article in articles:
            writer.writerow(article)

# 主程序
if __name__ == "__main__":
    all_articles = []
    for page in range(1, 6):  # 假设要爬取前5页
        articles = fetch_articles_from_page(page)
        all_articles.extend(articles)
    save_to_csv(all_articles, 'news_articles.csv')
    print("新闻数据已保存到 news_articles.csv")

代码解释:

  1. 错误处理 : 使用try-except块捕获请求异常,并打印错误信息。
  2. 重试机制 : 使用for循环和指数退避算法(time.sleep(2 ** attempt))实现重试机制。
定时任务

为了定期运行爬虫,可以使用系统的定时任务工具,如Linux的cron或Windows的任务计划程序。这里以cron为例,介绍如何定期运行爬虫。

步骤1:编写爬虫脚本

假设我们已经编写好了一个爬虫脚本news_spider.py

步骤2:配置 cron****任务

打开终端,输入crontab -e编辑定时任务。添加以下内容,每天凌晨2点运行爬虫脚本:

0 2 * * * /usr/bin/python3 /path/to/news_spider.py

代码解释:

  1. 定时配置 : 0 2 * * *表示每天凌晨2点运行。
  2. 运行脚本: 指定Python解释器和爬虫脚本的路径。
性能优化

为了提高爬虫的性能和效率,可以采用以下优化策略:

  1. 并发和多线程: 使用多线程或异步编程加速爬取速度。
  2. 减少重复请求: 使用缓存或数据库存储已爬取的URL,避免重复请求。
  3. 优化解析速度 : 使用更高效的HTML解析库,如lxml

示例:使用多线程优化爬虫

python 复制代码
import concurrent.futures
import requests
from bs4 import BeautifulSoup
import csv

# 文章列表页URL模板
base_url = "http://news.example.com/page/"
max_workers = 5  # 最大线程数

# 爬取文章详情的函数
def fetch_article(url):
    try:
        response = requests.get(url)
        response.raise_for_status()
        soup = BeautifulSoup(response.content, 'html.parser')
        title = soup.find('h1', class_='article-title').text
        author = soup.find('span', class_='article-author').text
        date = soup.find('span', class_='article-date').text
        content = soup.find('div', class_='article-content').text
        return {
            'title': title,
            'author': author,
            'date': date,
            'content': content
        }
    except requests.exceptions.RequestException as e:
        print(f"请求失败: {e}")
        return None

# 爬取文章列表页的函数
def fetch_articles_from_page(page):
    url = f"{base_url}{page}"
    try:
        response = requests.get(url)
        response.raise_for_status()
        soup = BeautifulSoup(response.content, 'html.parser')
        links = soup.find_all('a', class_='article-link')
        article_urls = [link['href'] for link in links]
        return article_urls
    except requests.exceptions.RequestException as e:
        print(f"请求失败: {e}")
        return []

# 主程序
if __name__ == "__main__":
    all_articles = []
    with concurrent.futures.ThreadPoolExecutor(max_workers=max_workers) as executor:
        # 爬取前5页的文章URL
        article_urls = []
        for page in range(1, 6):
            article_urls.extend(fetch_articles_from_page(page))
        
        # 并发爬取文章详情
        future_to_url = {executor.submit(fetch_article, url): url for url in article_urls}
        for future in concurrent.futures.as_completed(future_to_url):
            article = future.result()
            if article:
                all_articles.append(article)

    # 保存数据到CSV文件
    save_to_csv(all_articles, 'news_articles.csv')
    print("新闻数据已保存到 news_articles.csv")

代码解释:

  1. 并发爬取文章详情 : 使用concurrent.futures.ThreadPoolExecutor实现多线程并发爬取文章详情。
  2. 优化爬取速度: 使用多线程提高爬取速度。
结论

通过错误处理与重试机制、定时任务和性能优化,可以显著提高爬虫的稳定性和效率。本文详细介绍了这些维护与优化技术,帮助我们编写高效稳定的爬虫程序。

相关推荐
ℳ₯㎕ddzོꦿ࿐2 小时前
解决Python 在 Flask 开发模式下定时任务启动两次的问题
开发语言·python·flask
CodeClimb2 小时前
【华为OD-E卷 - 第k个排列 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
一水鉴天2 小时前
为AI聊天工具添加一个知识系统 之63 详细设计 之4:AI操作系统 之2 智能合约
开发语言·人工智能·python
Channing Lewis2 小时前
什么是 Flask 的蓝图(Blueprint)
后端·python·flask
B站计算机毕业设计超人2 小时前
计算机毕业设计hadoop+spark股票基金推荐系统 股票基金预测系统 股票基金可视化系统 股票基金数据分析 股票基金大数据 股票基金爬虫
大数据·hadoop·python·spark·课程设计·数据可视化·推荐算法
觅远3 小时前
python+playwright自动化测试(四):元素操作(键盘鼠标事件)、文件上传
python·自动化
ghostwritten3 小时前
Python FastAPI 实战应用指南
开发语言·python·fastapi
CM莫问4 小时前
python实战(十五)——中文手写体数字图像CNN分类
人工智能·python·深度学习·算法·cnn·图像分类·手写体识别
通信.萌新5 小时前
OpenCV边沿检测(Python版)
人工智能·python·opencv
Bran_Liu5 小时前
【LeetCode 刷题】字符串-字符串匹配(KMP)
python·算法·leetcode