0

0

使用Selenium从Google地图提取商家评分和评论数

DDD

DDD

发布时间:2025-07-16 19:32:02

|

774人浏览过

|

来源于php中文网

原创

使用Selenium从Google地图提取商家评分和评论数

本文详细介绍了如何使用Selenium库从Google地图搜索结果中高效地提取商家评分和评论数量。教程涵盖了Selenium环境配置、动态页面滚动加载更多结果的策略、以及关键的元素定位技巧,特别是针对Google地图动态内容中评分和评论的准确XPath定位。通过示例代码和最佳实践,帮助读者掌握从复杂Web应用中抓取数据的专业方法。

1. 环境准备与Selenium基础配置

在开始之前,请确保您的python环境中已安装selenium库,并下载了与您的chrome浏览器版本兼容的chromedriver。

首先,导入必要的Selenium模块,并进行基本的WebDriver配置。Options().add_experimental_option("detach", True) 选项可以使浏览器在脚本执行完毕后不立即关闭,方便调试。

from selenium import webdriver
from selenium.webdriver.chrome.options import Options
from selenium.webdriver.common.by import By
from selenium.webdriver.common.keys import Keys
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver import ActionChains
from selenium.webdriver.support.ui import WebDriverWait
import time

# 配置Chrome选项,使其在脚本结束后保持打开
chrome_options = Options()
chrome_options.add_experimental_option("detach", True)

# 初始化WebDriver、ActionChains和WebDriverWait
driver = webdriver.Chrome(options=chrome_options)
actionChains = ActionChains(driver)
wait = WebDriverWait(driver, 20)

# 定义一个辅助函数,用于等待元素位置稳定
def wait_for_element_location_to_be_stable(element):
    """
    等待一个元素的位置在一段时间内保持不变,以确保页面加载稳定。
    这对于动态加载的元素特别有用。
    """
    initial_location = element.location
    previous_location = initial_location
    start_time = time.time()
    while time.time() - start_time < 1: # 等待1秒钟,期间位置未变化则认为稳定
        current_location = element.location
        if current_location != previous_location:
            previous_location = current_location
            start_time = time.time() # 位置变化,重置计时器
        time.sleep(0.4) # 短暂等待,避免CPU占用过高

# 导航到Google主页并接受Cookie(如果弹出)
driver.get("https://www.google.com/")
try:
    # 尝试点击Google的Cookie同意按钮
    cookie_accept_button = wait.until(EC.element_to_be_clickable((By.ID, "L2AGLb")))
    cookie_accept_button.click()
except:
    print("未找到或无需点击Cookie同意按钮。")

# 导航到Google地图
driver.get("https://www.google.com/maps")

# 等待页面加载并输入搜索查询
time.sleep(3) # 给予页面足够时间加载
search_box = wait.until(EC.presence_of_element_located((By.ID, "searchboxinput")))
search_box.send_keys("jardins in toulouse")
search_box.send_keys(Keys.RETURN)

# 等待搜索结果加载
time.sleep(5)

2. 动态加载结果的滚动机制

Google地图的搜索结果通常是动态加载的,这意味着初次加载时可能只显示一部分结果。为了获取更多结果,需要模拟用户向下滚动的行为。本教程采用了一种鲁棒的滚动策略,通过滚动到列表中的最后一个元素并发送“向下箭头”键来触发新内容的加载,并判断滚动是否停止(即没有新内容加载出来)。

# 获取初始结果列表
# 这里的'hfpxzc'是Google地图中每个商家/地点的链接元素的通用类名
results = wait.until(EC.presence_of_all_elements_located((By.XPATH, "//a[@class='hfpxzc']")))

break_condition = False
# 找到一个可以聚焦的元素,通常是搜索框或者地图上的某个固定元素
# 这里使用'zero-input'作为示例,实际中可能需要根据页面结构调整
focus_element = driver.find_element(By.ID, 'zero-input')

while not break_condition:
    # 记录当前列表的最后一个元素
    temp_last_element = results[-1]

    # 滚动到最后一个元素,确保其在视图中
    actionChains.scroll_to_element(results[-1]).perform()

    # 将焦点移到某个元素并点击,然后发送向下箭头键,确保滚动事件被触发
    # 这一步是为了确保滚动条或页面本身响应键盘事件
    actionChains.move_to_element(focus_element).click().perform()
    for _ in range(3): # 多次发送向下箭头键,确保滚动足够
        actionChains.send_keys(Keys.ARROW_DOWN).perform()
        time.sleep(0.5) # 短暂暂停以允许页面响应

    # 等待最后一个元素的位置稳定,这有助于判断页面是否已停止加载
    wait_for_element_location_to_be_stable(temp_last_element)

    # 重新获取所有结果,检查是否有新元素加载
    results = wait.until(EC.presence_of_all_elements_located((By.XPATH, "//a[@class='hfpxzc']")))

    # 如果新的结果列表的最后一个元素与之前的最后一个元素相同,
    # 并且列表长度不再增加,则认为已滚动到底部
    if results[-1] == temp_last_element and len(results) > len(set(results)): # 简单的去重判断,确保不是重复元素
        break_condition = True
    elif len(results) >= 100: # 如果已经获取了足够数量的结果,也可以停止
        break_condition = True

注意: 上述滚动逻辑中的 len(results) > len(set(results)) 检查是为了更严谨地判断是否加载了新元素,因为 results[-1] == temp_last_element 可能在某些情况下误判。更可靠的方法是比较新旧列表的长度,或者检查新列表中是否包含旧列表的所有元素且有新增。此处简化为当最后一个元素相同时,假设已到底部,或达到目标数量。

3. 精准提取商家评分和评论数

这是解决原始问题的核心部分。原始代码使用了一个绝对XPath,这在迭代多个结果时是无效的。正确的做法是使用相对XPath,从当前迭代的商家元素(result)出发来定位其相关的评分和评论元素。

MagickPen
MagickPen

在线AI英语写作助手,像魔术师一样在几秒钟内写出任何东西。

下载

Google地图中,每个商家/地点的信息通常在一个“卡片容器”内。result 变量(即 //a[@class='hfpxzc'])是这个卡片容器内的链接元素。评分(例如“4.1”)通常在一个具有特定类名(如 MW4etd)的 span 元素中,而评论数量(例如“(123)”)则可能紧邻评分,或者在另一个具有特定属性的 span 元素中。

定位策略:

  1. 从 result 元素向上移动到其父级(..),因为评分和评论通常不是 hfpxzc 链接的直接子元素。
  2. 在父级或其子孙元素中,寻找具有特定类名或文本模式的元素。
    • 评分:通常是 //*[@class='MW4etd']。
    • 评论数:通常是包含括号数字的文本,例如 //span[contains(text(), '(') and contains(text(), ')')]。
# 循环遍历结果并提取名称、评分和评论数
# 限制在获取前100个结果(如果存在)
for i, result in enumerate(results[:100], start=1):
    name = result.get_attribute('aria-label')

    # 提取评分
    # 评分通常在当前商家链接元素的父级容器内,具有'MW4etd'类
    ratings_elements = result.find_elements(By.XPATH, "..//*[@class='MW4etd']")
    rating = ratings_elements[0].text if len(ratings_elements) > 0 else "评分不可用"

    # 提取评论数量
    # 评论数量通常是包含在括号内的数字,与评分元素相邻或在同一父级下
    # 这里使用一个更通用的XPath来查找包含括号的文本,以适应不同的HTML结构
    review_count_elements = result.find_elements(By.XPATH, "..//span[contains(text(), '(') and contains(text(), ')')]")
    review_count = review_count_elements[0].text if len(review_count_elements) > 0 else "评论数不可用"

    print(f"结果 {i}: 名称: {name} - 评分: {rating} - 评论数: {review_count}")

# 关闭浏览器
driver.quit()

4. 注意事项与最佳实践

  • XPath的鲁棒性: 避免使用绝对XPath,因为它们对页面结构的变化非常敏感。优先使用相对XPath (. 或 .. 开头) 或 CSS 选择器,结合元素的类名、ID、文本内容或属性来定位。
  • 等待策略: 动态网页内容需要强大的等待机制。WebDriverWait 结合 expected_conditions 比简单的 time.sleep() 更可靠,因为它会等待直到条件满足或超时。wait_for_element_location_to_be_stable 这样的自定义等待函数对于处理元素移动或重绘的情况非常有用。
  • 错误处理: 使用 find_elements 而不是 find_element 来查找可能不存在的元素。find_elements 会返回一个列表,如果元素不存在则返回空列表,这允许你通过检查列表长度来避免 NoSuchElementException。
  • 模拟真实用户行为: 模拟鼠标点击、键盘输入和滚动等操作,可以减少被网站识别为自动化脚本的风险。
  • 效率与性能: 频繁的 time.sleep() 会降低脚本执行效率。在确保页面加载完成的前提下,尽量减少不必要的等待时间。
  • 反爬机制: Google地图等大型网站有复杂的反爬机制。频繁或大量的请求可能导致IP被封禁或需要验证码。考虑使用代理IP、降低请求频率、模拟用户代理等策略。
  • 道德与法律: 在进行任何形式的网络抓取之前,请务必阅读并遵守目标网站的服务条款和隐私政策。尊重网站的数据所有权,并确保您的抓取行为合法合规。

总结

本教程提供了一个使用Selenium从Google地图提取商家评分和评论数的完整解决方案。通过理解动态内容加载机制和掌握正确的元素定位策略,您可以有效地从复杂的Web应用中提取所需数据。记住,Web抓取是一个不断适应和学习的过程,随着网站结构的变化,您的抓取脚本也可能需要随之更新。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
chrome什么意思
chrome什么意思

chrome是浏览器的意思,由Google开发的网络浏览器,它在2008年首次发布,并迅速成为全球最受欢迎的浏览器之一。本专题为大家提供chrome相关的文章、下载、课程内容,供大家免费下载体验。

835

2023.08.11

chrome无法加载插件怎么办
chrome无法加载插件怎么办

chrome无法加载插件可以通过检查插件是否已正确安装、禁用和启用插件、清除插件缓存、更新浏览器和插件、检查网络连接和尝试在隐身模式下加载插件方法解决。更多关于chrome相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

744

2023.11.06

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

469

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

13

2025.12.06

class在c语言中的意思
class在c语言中的意思

在C语言中,"class" 是一个关键字,用于定义一个类。想了解更多class的相关内容,可以阅读本专题下面的文章。

469

2024.01.03

python中class的含义
python中class的含义

本专题整合了python中class的相关内容,阅读专题下面的文章了解更多详细内容。

13

2025.12.06

PHP 命令行脚本与自动化任务开发
PHP 命令行脚本与自动化任务开发

本专题系统讲解 PHP 在命令行环境(CLI)下的开发与应用,内容涵盖 PHP CLI 基础、参数解析、文件与目录操作、日志输出、异常处理,以及与 Linux 定时任务(Cron)的结合使用。通过实战示例,帮助开发者掌握使用 PHP 构建 自动化脚本、批处理工具与后台任务程序 的能力。

42

2025.12.13

java入门学习合集
java入门学习合集

本专题整合了java入门学习指南、初学者项目实战、入门到精通等等内容,阅读专题下面的文章了解更多详细学习方法。

1

2026.01.29

java配置环境变量教程合集
java配置环境变量教程合集

本专题整合了java配置环境变量设置、步骤、安装jdk、避免冲突等等相关内容,阅读专题下面的文章了解更多详细操作。

2

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.1万人学习

CSS教程
CSS教程

共754课时 | 24.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号