当前位置: 首页 > news >正文

Ruby爬虫如何控制并发数量:爬取京东电子产品

1. 引言

京东作为中国最大的电商平台之一,拥有海量的商品信息,其中电子产品是其热门品类之一。对于市场研究人员、数据分析师和开发者来说,能够高效地爬取和分析这些数据具有重要的价值。然而,京东网站的复杂性和反爬措施使得爬取任务更具挑战性。合理控制并发数量不仅能提高爬取效率,还能避免触发网站的反爬机制,确保爬虫的稳定运行。

Ruby语言以其简洁的语法和强大的并发处理能力,结合Nokogiri等库,成为开发高效爬虫的理想选择。本文将通过一个实战案例,展示如何在Ruby爬虫中控制并发数量,同时解析京东电子产品页面并提取标题。

2. 爬虫设计与实现

在本节中,我们将详细介绍如何设计和实现一个Ruby爬虫,用于爬取京东电子产品页面并提取标题。我们将从简单的单线程爬虫开始,逐步扩展到并发爬虫,并最终实现动态并发控制。

2.1 单线程爬虫

单线程爬虫是最基本的爬虫形式,适用于小规模数据爬取。以下是单线程爬虫的实现代码:

require 'nokogiri'
require 'open-uri'

proxy_host = "www.16yun.cn"
proxy_port = 5445
proxy_user = "16QMSOML"
proxy_pass = "280651"

def fetch_title(url, proxy_host, proxy_port, proxy_user, proxy_pass)
  begin
    # 使用代理服务器打开网页
    doc = Nokogiri::HTML(URI.open(url, proxy_addr: proxy_host, proxy_port: proxy_port,
                                  proxy_user: proxy_user, proxy_pass: proxy_pass))
    doc.at('title').text
  rescue StandardError => e
    puts "Error fetching #{url}: #{e.message}"
    nil
  end
end

# 示例:爬取京东电子产品首页
url = 'https://www.jd.com/category/1320-13753-13883.html'
title = fetch_title(url, proxy_host, proxy_port, proxy_user, proxy_pass)
puts "Title: #{title}"

2.2 并发爬虫

单线程爬虫效率较低,尤其是在爬取大量页面时。通过并发处理,可以显著提高爬取效率。以下是使用concurrent-ruby库实现的并发爬虫代码:

require 'nokogiri'
require 'open-uri'
require 'concurrent'

proxy_host = "www.16yun.cn"
proxy_port = 5445
proxy_user = "16QMSOML"
proxy_pass = "280651"

def fetch_title(url, proxy_host, proxy_port, proxy_user, proxy_pass)
  begin
    doc = Nokogiri::HTML(URI.open(url, proxy_addr: proxy_host, proxy_port: proxy_port,
                                  proxy_user: proxy_user, proxy_pass: proxy_pass))
    doc.at('title').text
  rescue StandardError => e
    puts "Error fetching #{url}: #{e.message}"
    nil
  end
end

# 创建线程池,最大并发数为5
pool = Concurrent::ThreadPoolExecutor.new(max_threads: 5)

urls = [
  'https://www.jd.com/category/1320-13753-13883.html', # 电子产品
  'https://www.jd.com/category/1320-13754-13884.html', # 手机
  'https://www.jd.com/category/1320-13755-13885.html'  # 笔记本电脑
]

# 使用线程池并发爬取
results = urls.map do |url|
  pool.post(url, proxy_host, proxy_port, proxy_user, proxy_pass) do |u, host, port, user, pass|
    fetch_title(u, host, port, user, pass)
  end
end

titles = results.map(&:value).compact
titles.each { |title| puts "Title: #{title}" }

pool.shutdown

2.3 动态并发调整

在实际应用中,目标网站的响应时间可能因网络状况和服务器负载而波动。通过动态调整并发数量,可以优化爬取效率,同时避免对目标网站造成过大压力。以下是动态并发调整的实现代码:

require 'nokogiri'
require 'open-uri'
require 'concurrent'

def fetch_title(url, pool)
  start_time = Time.now
  begin
    doc = Nokogiri::HTML(URI.open(url))
    title = doc.at('title').text
    puts "Fetched #{url}: #{title}"
    title
  rescue StandardError => e
    puts "Error fetching #{url}: #{e.message}"
    nil
  ensure
    elapsed_time = Time.now - start_time
    adjust_concurrency(pool, elapsed_time)
  end
end

def adjust_concurrency(pool, elapsed_time)
  if elapsed_time < 0.5
    pool.resize([pool.max_threads + 1, 10].min) # 最大并发数不超过10
  elsif elapsed_time > 2.0
    pool.resize([pool.max_threads - 1, 1].max) # 最小并发数为1
  end
end

pool = Concurrent::ThreadPoolExecutor.new(max_threads: 5)
urls = [
  'https://www.jd.com/category/1320-13753-13883.html',
  'https://www.jd.com/category/1320-13754-13884.html',
  'https://www.jd.com/category/1320-13755-13885.html'
]

results = urls.map { |url| pool.post(url, pool, &method(:fetch_title)) }
titles = results.map(&:value).compact

titles.each { |title| puts "Title: #{title}" }

pool.shutdown

总结

本文通过一个具体的实战案例——爬取京东电子产品页面并提取标题,详细介绍了如何在Ruby爬虫中控制并发数量。从基础的单线程爬虫到并发爬虫,再到动态并发调整和分布式爬虫,本文为读者提供了一个全面的指南。通过合理控制并发数量,不仅可以提高爬取效率,还可以避免触发目标网站的反爬机制,确保爬虫的稳定运行。

相关文章:

  • 在 Docker 中,无法直接将外部多个端口映射到容器内部的同一个端口
  • 前端多角色权限页面(同浏览器同时登录)数据互串解决
  • 常见的限流算法有哪些
  • Redis 面试专题
  • vue2.6附件预览及下载
  • Python的那些事第四十一篇:简化数据库交互的利器Django ORM
  • HttpRunner框架 核心知识的系统复习
  • linux磁盘非lvm分区
  • 【高并发】消息队列(MQ)全解析:原理、主流产品及 Java 实现
  • 自由学习记录(42)
  • 基于SNR估计的自适应码率LDPC编译码算法matlab性能仿真,对比固定码率LDPC的系统传输性能
  • Golang的数据库分库分表策略
  • llamafactory大模型微调教程(周易大模型案例)
  • 解锁STM32外设:开启嵌入式开发新世界
  • ⭐算法OJ⭐跳跃游戏【BFS+滑动窗口】(C++实现)Jump Game 系列 III,VII
  • C++智能指针shared_ptr
  • 从0开始的操作系统手搓教程23:构建输入子系统——实现键盘驱动1——热身驱动
  • OpenGL ES -> GLSurfaceView纹理贴图VBO(Vertex Buffer Object)方法实现
  • 人工智能+乡村振兴+文旅+低空无人机产业链技术详解
  • MySQL5.7.44-winx64版本Windows Server下载安装教程图解
  • 网站建设管理与维护/2023重大新闻事件10条
  • 营销型网站建设方案演讲ppt/软文代写平台
  • dnf交易网站建设/百度推广怎么登陆
  • 建设交通人才网站/山东百度推广
  • 网站设计优化方案/搜索百度
  • 那个网站可以做学历认证/郑州seo排名工具