Python 网络爬虫进阶:动态网页爬取与反爬机制应对

news2024/11/28 5:43:55

在上一篇文章中,我们学习了如何使用 Python 构建一个基本的网络爬虫。然而,在实际应用中,许多网站使用动态内容加载或实现反爬机制来阻止未经授权的抓取。因此,本篇文章将深入探讨以下进阶主题:

  • 如何处理动态加载的网页内容
  • 应对常见的反爬机制
  • 爬虫性能优化

通过具体实例,我们将探讨更复杂的网络爬虫开发技巧。


一、动态网页爬取

现代网页通常通过 JavaScript 加载动态内容。直接使用 requests 获取的 HTML 可能不包含目标数据。这时,我们可以使用 selenium 模拟浏览器行为来抓取动态内容。


1. 安装与配置 Selenium

安装 Selenium 和浏览器驱动
pip install selenium

下载对应浏览器的驱动程序(如 ChromeDriver),并将其路径添加到系统变量中。

初始化 WebDriver

以下是一个基本的 Selenium 配置:

from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.chrome.service import Service
from selenium.webdriver.common.keys import Keys
import time

# 设置浏览器驱动路径
driver_path = "path/to/chromedriver"
service = Service(driver_path)

# 初始化 WebDriver
driver = webdriver.Chrome(service=service)

# 打开网页
driver.get("http://quotes.toscrape.com/js/")
time.sleep(2)  # 等待动态内容加载

# 获取网页内容
print(driver.page_source)

# 关闭浏览器
driver.quit()

2. 爬取动态加载的名言

通过 Selenium 提取动态内容:

# 初始化 WebDriver
driver = webdriver.Chrome(service=Service("path/to/chromedriver"))
driver.get("http://quotes.toscrape.com/js/")

# 等待内容加载
time.sleep(2)

# 提取名言和作者
quotes = driver.find_elements(By.CLASS_NAME, "quote")
for quote in quotes:
    text = quote.find_element(By.CLASS_NAME, "text").text
    author = quote.find_element(By.CLASS_NAME, "author").text
    print(f"名言: {text}\n作者: {author}\n")

driver.quit()

3. 处理翻页

动态页面通常通过点击“下一页”按钮加载更多内容。我们可以模拟用户操作实现翻页爬取。

# 自动翻页爬取
while True:
    quotes = driver.find_elements(By.CLASS_NAME, "quote")
    for quote in quotes:
        text = quote.find_element(By.CLASS_NAME, "text").text
        author = quote.find_element(By.CLASS_NAME, "author").text
        print(f"名言: {text}\n作者: {author}\n")

    # 点击下一页按钮
    try:
        next_button = driver.find_element(By.CLASS_NAME, "next")
        next_button.click()
        time.sleep(2)  # 等待页面加载
    except:
        print("已到最后一页")
        break

driver.quit()

二、应对反爬机制

很多网站会通过检测 IP、User-Agent 或频繁访问行为来阻止爬虫。以下是一些常见反爬机制和应对策略:


1. 模拟真实用户行为

设置请求头

在 HTTP 请求中,添加 User-Agent 模拟浏览器。

headers = {
    "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/114.0.5735.199 Safari/537.36"
}

response = requests.get("http://example.com", headers=headers)
随机延迟

为避免触发频率限制,爬取时可以随机添加延迟。

import time
import random

time.sleep(random.uniform(1, 3))  # 随机延迟 1 到 3 秒

2. 使用代理 IP

通过代理 IP 隐藏爬虫的真实 IP,防止被封禁。

proxies = {
    "http": "http://your_proxy:port",
    "https": "http://your_proxy:port"
}

response = requests.get("http://example.com", proxies=proxies)
自动获取免费代理

可以使用爬虫定期抓取免费代理(如西刺代理),并动态切换 IP。


3. 验证码处理

部分网站使用验证码拦截爬虫。应对策略包括:

  • 手动输入:提示用户输入验证码。
  • 验证码识别服务:如 平台 提供的 API。
  • 避开验证码:尝试通过 API 或其他无需验证码的接口获取数据。

三、性能优化

当需要爬取大量数据时,爬虫的性能优化尤为重要。


1. 多线程或多进程

使用多线程或多进程提高爬取效率:

多线程示例
import threading

def fetch_data(url):
    response = requests.get(url)
    print(f"{url} 完成")

urls = ["http://example.com/page1", "http://example.com/page2"]

threads = []
for url in urls:
    thread = threading.Thread(target=fetch_data, args=(url,))
    threads.append(thread)
    thread.start()

for thread in threads:
    thread.join()

2. 异步爬取

使用 aiohttpasyncio 实现异步爬取。

import aiohttp
import asyncio

async def fetch(session, url):
    async with session.get(url) as response:
        return await response.text()

async def main():
    urls = ["http://example.com/page1", "http://example.com/page2"]
    async with aiohttp.ClientSession() as session:
        tasks = [fetch(session, url) for url in urls]
        results = await asyncio.gather(*tasks)
        for content in results:
            print(content)

asyncio.run(main())

3. 数据去重

避免重复爬取相同数据,可以使用哈希或数据库记录已访问 URL。

visited = set()

if url not in visited:
    visited.add(url)
    # 执行爬取

四、实战案例:动态商品价格爬取

以下示例演示如何抓取电商网站动态加载的商品价格,并应对翻页和反爬机制。

from selenium import webdriver
from selenium.webdriver.common.by import By
import time

driver = webdriver.Chrome(service=Service("path/to/chromedriver"))
driver.get("https://example-ecommerce.com")

# 等待加载
time.sleep(3)

# 爬取商品信息
while True:
    items = driver.find_elements(By.CLASS_NAME, "product-item")
    for item in items:
        name = item.find_element(By.CLASS_NAME, "product-title").text
        price = item.find_element(By.CLASS_NAME, "product-price").text
        print(f"商品: {name}, 价格: {price}")

    # 尝试翻页
    try:
        next_button = driver.find_element(By.CLASS_NAME, "next-page")
        next_button.click()
        time.sleep(2)  # 等待页面加载
    except:
        print("已到最后一页")
        break

driver.quit()

五、总结

通过本篇文章,你学习了以下进阶爬虫技巧:

  1. 使用 Selenium 处理动态网页。
  2. 应对常见反爬机制,如设置代理、随机延迟等。
  3. 提升爬取性能的方法,包括多线程和异步爬取。

下一步,建议尝试构建一个完整的爬虫项目,如爬取新闻网站、商品价格监控等,并学习如何处理复杂的反爬场景。祝你爬虫之路越走越远!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2248851.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何借助AI生成PPT,让创作轻松又高效

PPT是现代职场中不可或缺的表达工具,但同时也可能是令人抓狂的时间杀手。几页幻灯片的制作,常常需要花费数小时调整字体、配色与排版。AI的飞速发展为我们带来了革新——AI生成PPT的技术不仅让制作流程大大简化,还重新定义了效率与创意的关系…

【Linux】Make/Makefile

这个3/4行的语法和1/2行是一样的。也是依赖关系和依赖方法。 make命令扫描makefile文件时,从上向下扫描,默认形成一个目标文件。 指定make clean的时候才回去执行对应的清除。 为什么要给我们的clean.PHONY:clean声明它是伪目标呢? PHONY类…

HarmonyOS:@Provide装饰器和@Consume装饰器:与后代组件双向同步

一、前言 Provide和Consume,应用于与后代组件的双向数据同步,应用于状态数据在多个层级之间传递的场景。不同于上文提到的父子组件之间通过命名参数机制传递,Provide和Consume摆脱参数传递机制的束缚,实现跨层级传递。 其中Provi…

如何做好一份技术文档?

打造出色技术文档的艺术 在当今技术驱动的世界中,技术文档扮演着至关重要的角色。它不仅是工程师和开发人员之间交流的桥梁,更是产品和技术成功的隐形推手。一份优秀的技术文档宛如一张精准的航海图,能够引导读者穿越技术的迷雾,…

泰山众筹怎样吸引用户参与

泰山众筹项目要吸引用户参与,需要采取一系列策略来增强项目的吸引力、提高用户信任度,并激发用户的参与热情。以下是一些建议: 1. 明确项目价值与愿景 展示独特性:明确泰山众筹项目的独特卖点,如创新性、社会影响力或…

抓包之验证content-length响应头的作用

写在前面 根据http协议的规范,content-length响应头用来标记固定长度响应信息长度,http客户端,比如浏览器也会解析这个字段来进行数据的解析。 1:测试 1.1:content-length等于实际内容匹配时 使用python脚本testco…

T3 TensorFlow入门实战——天气识别

🍨 本文為🔗365天深度學習訓練營 中的學習紀錄博客🍖 原作者:K同学啊 | 接輔導、項目定制 一、前期准备 1. 导入数据 # Import the required libraries import numpy as np import os,PIL,pathlib import matplotlib.pyplot as …

✨系统设计时应时刻考虑设计模式基础原则

目录 💫单一职责原则 (Single Responsibility Principle, SRP)💫开放-封闭原则 (Open-Closed Principle, OCP)💫依赖倒转原则 (Dependency Inversion Principle, DIP)💫里氏代换原则 (Liskov Substitution Principle, LSP)&#x…

fatal error in include chain (rtthread.h):rtconfig.h file not found

项目搜索这个文件 rtconfig 找到后将其复制粘贴到 你的目录\Keil\ARM\ARMCC\include 应该还有cJSON,rtthread.h和 等也复制粘贴下

【回文数组——另类递推】

题目 代码 #include <bits/stdc.h> using namespace std; using ll long long; const int N 1e510; int a[N], b[N]; int main() {int n;cin >> n;for(int i 1; i < n; i)cin >> a[i];for(int i 1; i < n / 2; i)b[i] a[i] - a[n1-i];ll ans 0;…

SQL基础入门—— 简单查询与条件筛选

在SQL中&#xff0c;查询是从数据库中获取数据的核心操作&#xff0c;而条件筛选是查询中不可或缺的一部分。通过使用条件筛选&#xff0c;我们可以精准地从大量数据中提取我们需要的信息。本节将详细讲解如何使用SQL进行简单查询与条件筛选&#xff0c;包含常见的条件运算符和…

反向代理模块

1 概念 1.1 反向代理概念 反向代理是指以代理服务器来接收客户端的请求&#xff0c;然后将请求转发给内部网络上的服务器&#xff0c;将从服务器上得到的结果返回给客户端&#xff0c;此时代理服务器对外表现为一个反向代理服务器。 对于客户端来说&#xff0c;反向代理就相当于…

英语知识网站:Spring Boot技术构建

6系统测试 6.1概念和意义 测试的定义&#xff1a;程序测试是为了发现错误而执行程序的过程。测试(Testing)的任务与目的可以描述为&#xff1a; 目的&#xff1a;发现程序的错误&#xff1b; 任务&#xff1a;通过在计算机上执行程序&#xff0c;暴露程序中潜在的错误。 另一个…

TMS FNC UI Pack 5.4.0 for Delphi 12

TMS FNC UI Pack是适用于 Delphi 和 C Builder 的多功能 UI 控件的综合集合&#xff0c;提供跨 VCL、FMX、LCL 和 TMS WEB Core 等平台的强大功能。这个统一的组件集包括基本工具&#xff0c;如网格、规划器、树视图、功能区和丰富的编辑器&#xff0c;确保兼容性和简化的开发。…

【AIGC】国内AI工具复现GPTs效果详解

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: AIGC | GPTs应用实例 文章目录 &#x1f4af;前言&#x1f4af;本文所要复现的GPTs介绍&#x1f4af;GPTs指令作为提示词在ChatGPT实现类似效果&#x1f4af;国内AI工具复现GPTs效果可能出现的问题解决方法解决后的效…

网络原理(一):应用层自定义协议的信息组织格式 HTTP 前置知识

目录 1. 应用层 2. 自定义协议 2.1 根据需求 > 明确传输信息 2.2 约定好信息组织的格式 2.2.1 行文本 2.2.2 xml 2.2.3 json 2.2.4 protobuf 3. HTTP 协议 3.1 特点 4. 抓包工具 1. 应用层 在前面的博客中, 我们了解了 TCP/IP 五层协议模型: 应用层传输层网络层…

【es6】原生js在页面上画矩形及删除的实现方法

画一个矩形&#xff0c;可以选中高亮&#xff0c;删除自己效果的实现&#xff0c;后期会丰富下细节&#xff0c;拖动及拖动调整矩形大小 实现效果 代码实现 class Draw {constructor() {this.x 0this.y 0this.disX 0this.disY 0this.startX 0this.startY 0this.mouseDo…

12 —— Webpack中向前端注入环境变量

环境变量的作用&#xff1a;根据不同环境&#xff0c;执行不同的配置 需求&#xff1a;开发模式下打印语句生效&#xff0c;生产模式下打印语句失效 —— 使用Webpack内置的DefinePlugin插件 const webpack require(webpack) module.exports { plugins: [ new webpack.Def…

Learn Git Branching 学习笔记

网址&#xff1a;Learn Git Branching 一、基础篇 1.1 git commit 1.1.1 示例&#xff08;git commit&#xff09; git commit 1.1.2 题目&#xff08;两次提交记录&#xff09; git commit git commit 前 后 1.2 git branch 1.2.1 示例&#xff08;git branch <>、git …

Unity类银河战士恶魔城学习总结(P145 Save Skill Tree 保存技能树)

【Unity教程】从0编程制作类银河恶魔城游戏_哔哩哔哩_bilibili 教程源地址&#xff1a;https://www.udemy.com/course/2d-rpg-alexdev/ 本章节实现了技能树的保存 警告&#xff01;&#xff01;&#xff01; 如果有LoadData&#xff08;&#xff09;和SaveData(&#xff09;…