Основы парсинга сайтов: от HTML до готового датасета для NLP

Страницы:  1

Ответить
 

Professor Seleznov


Вы освоили машинное обучение, знаете, чем transformer отличается от LSTM, но где брать данные для своих проектов? Готовые датасеты — это хорошо, но они общие. А если вам нужны посты из конкретного Telegram‑канала, отзывы с узкопрофильного форума или корпус текстов по редкой теме?
Парсинг сайтов — это навык, который превращает интернет в вашу персональную фабрику данных. Без него даже самая умная модель останется без «топлива». В этой статье мы рассмотрим весь путь: от первого запроса к сайту до готового датасета, пригодного для обучения.
Мы будем двигаться от простого к сложному. Вы узнаете, как парсить статические страницы, обходить базовые блокировки и организовывать масштабный сбор данных.
📍 Прежде чем переходить к практике, пройдите короткий бесплатный тест по NLP и языковым трансформерным моделям. Он поможет понять, в каких темах вы уже уверены, а что стоит подтянуть. ➞ Пройти тест

С чего всё начинается
Прежде чем писать код, нужно понять, как устроен веб. Представьте, что интернет — это огромная библиотека, а ваш браузер — библиотекарь, который ходит за книгами. Парсер — это робот‑библиотекарь, который делает это быстро и по заданному алгоритму.
Всё начинается с HTTP‑запроса. Когда вы вводите адрес сайта, ваш браузер отправляет серверу запрос:«Дай мне, пожалуйста, содержимое страницы». Сервер отвечает HTML‑кодом — это «скелет» страницы, размеченный тегами.
Пример простого HTML‑фрагмента:


   

Название товара


    1000 ₽
   

Описание товара...


Парсер должен найти этот блок (div class="product") и извлечь из него нужные данные: заголовок (h2), цену (span.price) и описание (p.description).
GET и POST являются двумя основными типами запросов. GET используется для получения данных. Когда вы открываете страницу товара, браузер отправляет GET‑запрос. В свою очередь POST используется для отправки данных на сервер, например, при заполнении формы поиска или авторизации. На начальном этапе вам почти всегда будет достаточно GET‑запросов.
Requests + BeautifulSoup
Это стандартная связка для парсинга, известная как «золотая середина» для большинства задач, потому что requests скачивает страницу, а BeautifulSoup помогает находить в ней данные.
Почему они хороши вместе? Requests выступает в роли курьера, доставляя вам HTML‑код, а BeautifulSoup — в роли аналитика, который этот код изучает и находит в нём закономерности. requests идеально подходит для базовых HTTP‑операций, а BeautifulSoup — для навигации по HTML‑дереву и поиска элементов.
Для установки необходимо выполнить следующее:
pip install requests beautifulsoup4 lxml
Здесь lxml — это быстрый парсер, который мы передадим BeautifulSoup для ускорения работы. Теперь давайте напишем скрипт, который загружает страницу и извлекает заголовки статей.
import requests
from bs4 import BeautifulSoup
# 1. Отправляем GET-запрос к сайту
url = "https://example.com/blog"
# Для реальных проектов всегда указывайте User-Agent, чтобы не походить на бота
headers = {
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
}
response = requests.get(url, headers=headers)
# 2. Проверяем, успешно ли прошёл запрос (код 200 означает "OK")
if response.status_code == 200:
# 3. Передаём HTML-код страницы BeautifulSoup для анализа
soup = BeautifulSoup(response.text, 'lxml')
# 4. Находим все элементы, содержащие заголовки.
# Предположим, заголовки статей находятся в теге

с классом "post-title"
    titles = soup.find_all('h2', class_='post-title')
    # 5. Извлекаем и выводим текст из каждого найденного элемента
    for title in titles:
        print(title.text.strip())
else:
    print(f"Ошибка загрузки страницы. Код: {response.status_code}")Краткий разбор полётов:

  • requests.get(url, headers=headers) загружает страницу.
  • BeautifulSoup(response.text, 'lxml') создаёт объект, с которым удобно работать.
  • soup.find_all('h2', class_='post-title')ищет все теги

    с классом post-title. Это один из самых мощных методов библиотеки.

  • title.text.strip() достаёт текст внутри тега и удаляет лишние пробелы и переносы строк.
Продвинутый поиск: CSS‑селекторы
Метод find_all — это конечно хорошо, но иногда удобнее использовать CSS‑селекторы. Это мощный язык запросов к элементам HTML. Давайте посмотрим несколько примеров использования.
Найдем все ссылки внутри элементов с классом «product»
links = soup.select('.product a')
Теперь поищем элемент с id=«main-content»
main_content = soup.select_one('#main-content')
Найдем все изображения внутри блока с классом «gallery»
images = soup.select('.gallery img')

Метод .select() возвращает список всех подходящих элементов, а.select_one() — первый подходящий. Это делает код более читаемым, особенно при работе со сложной вложенностью.
Когда страницы оживают: Playwright для SPA и React‑сайтов
Современный интернет — это уже не просто набор HTML‑страниц. Многие сайты (особенно с технологиями React, Vue, Angular) — это одностраничные приложения (SPA). Они загружают пустой «скелет», а весь контент подтягивается скриптами уже после загрузки.
Вы можете получить HTML‑код такой страницы через requests, но красивого текста с данными в нём не будет. Вам нужен инструмент, который может не просто скачать файл, а запустить браузер и выполнить весь JavaScript‑код, как это делает обычный пользователь. И здесь на сцену выходит Playwright.
Playwright — это библиотека для автоматизации браузеров (Chrome, Firefox, Safari). Она запускает настоящее браузерное окно (или «голову», headless‑режим) и выполняет все действия, которые может делать человек: кликать, скроллить, вводить текст. После того как страница полностью загружена и все скрипты отработали, мы можем получить её финальный HTML‑код и спокойно распарсить его уже знакомым BeautifulSoup. Это незаменимый инструмент для сбора данных с SPA, сайтов с бесконечной прокруткой (infinite scroll) и для обхода несложной защиты от ботов.
Для установки выполним следующее:
pip install playwright
playwright install # Скачивает браузеры (chromium, firefox, webkit)

Далее рассмотрим пример скрипта для динамической страницы:
import asyncio
from playwright.async_api import async_playwright
from bs4 import BeautifulSoup

async def scrape_dynamic_site():
async with async_playwright() as p:
# Запускаем браузер (headless=False, чтобы увидеть окно браузера)
browser = await p.chromium.launch(headless=True)
page = await browser.new_page()

# Переходим на сайт и ждём, пока загрузится сеть (важно для SPA!)
await page.goto("https://quotes.toscrape.com/scroll")
await page.wait_for_load_state("networkidle")

# Симулируем прокрутку вниз для подгрузки новых элементов (если нужно)
for _ in range(3):
await page.evaluate("window.scrollTo(0, document.body.scrollHeight)")
await page.wait_for_timeout(1000) # Ждём подгрузки

# Получаем HTML-код полностью отрисованной страницы
html = await page.content()

# Закрываем браузер
await browser.close()

# Теперь можно парсить этот HTML через BeautifulSoup
soup = BeautifulSoup(html, 'lxml')
quotes = soup.select('.quote')

for quote in quotes:
text = quote.select_one('.text').text
author = quote.select_one('.author').text
print(f"{author} said: {text[:50]}...")
asyncio.run(scrape_dynamic_site())
В этом коде мы не просто скачиваем страницу, а эмулируем поведение пользователя: ждём загрузки, прокручиваем вниз, даём время подгрузиться новым элементам, и только потом забираем готовый HTML.
Пайплайн для качественного датасета
Получить сырой текст с сайта — это только полдела. Чтобы он превратился в полезный датасет для NLP, ему нужна обработка. Это похоже на добычу нефти: сначала идёт бурение (парсинг), а потом — сложная переработка (очистка), чтобы получить бензин (датасет).
Вот основные шаги по превращению «грязных» веб‑данных в пригодные для обучения модели:
  • Извлечение (Extraction): Мы уже это умеем. Получаем HTML и вытаскиваем из него нужные поля (текст, даты, теги).
  • Очистка (Cleaning): Удаляем HTML‑теги, которые могли остаться. Убираем мусор: специальные символы ( , \xa0, \u200b), лишние пробелы и переносы строк.
  • Нормализация (Normalization): Приводим текст к единому виду. Например, все даты приводим к формату ГГГГ‑ММ‑ДД, имена и названия очищаем от случайных символов.
Предобработка для NLP (Preprocessing): Этот шаг зависит от вашей задачи. Он может включать:
  • Токенизацию (разбиение на слова).
  • Удаление стоп‑слов (очень частых слов, не несущих смысла: «и», «в», «на»).
  • Лемматизацию или стемминг (приведение слов к начальной форме: «бежал», «бежит», «бегут» → «бежать»).
Теперь давайте рассмотрим пример кода, который дополняет наш пайплайн, превращая сырой HTML в чистый текст.
import re
import pandas as pd
from bs4 import BeautifulSoup
from nltk.corpus import stopwords
from nltk.stem import WordNetLemmatizer
# Не забудьте скачать ресурсы NLTK: nltk.download('stopwords'), nltk.download('wordnet')
lemmatizer = WordNetLemmatizer()
stop_words = set(stopwords.words('russian'))

def clean_text(html_content):
# 1. Извлечение текста из HTML
soup = BeautifulSoup(html_content, 'lxml')
text = soup.get_text(separator=' ')

# 2. Приведение к нижнему регистру и удаление спецсимволов
text = text.lower()
text = re.sub(r'[^а-яa-z0-9\s]', '', text) # Оставляем только буквы, цифры и пробелы

# 3. Токенизация (разбиение на слова)
tokens = text.split()

# 4. Удаление стоп-слов и коротких слов
tokens = [token for token in tokens if token not in stop_words and len(token) > 2]

# 5. Лемматизация
tokens = [lemmatizer.lemmatize(token) for token in tokens]

# 6. Сборка обратно в строку
clean_text = ' '.join(tokens)
return clean_text
# Пример использования в вашем пайплайне
raw_data = [] # список словарей, полученных после парсинга
for item in raw_data:
item['cleaned_content'] = clean_text(item['raw_html'])
df = pd.DataFrame(raw_data)
df.to_csv('my_dataset.csv', index=False) # Сохраняем готовый датасет
Следуя этому пайплайну, вы превратите ворох сырых HTML‑страниц в структурированный, чистый датасет, готовый для обучения моделей классификации, генерации текста или построения RAG‑систем.
Заключение
Парсинг сайтов — это фундаментальный навык для любого специалиста по данным, который хочет работать с реальными, неигрушечными задачами. Мы прошли путь от базовых HTTP‑запросов до запуска браузеров и очистки данных. Теперь у вас есть инструментарий, чтобы создавать собственные датасеты для самых смелых NLP‑проектов.
pic
Когда нужно не просто «поиграться с LLM», а встроить языковые модели в реальный продукт, быстро выясняется: базового понимания ChatGPT уже недостаточно. Нужно разбираться, как работают трансформеры, embeddings, RAG, тонкая настройка моделей и оценка качества, иначе проект легко превращается в набор красивых демо без стабильного результата.
Курс «Языковые трансформенные модели / NLP» помогает перейти от поверхностного использования нейросетей к инженерному пониманию NLP‑систем: как выбирать подходящую архитектуру, готовить данные, работать с текстовыми корпусами и собирать решения, которые можно применять в рабочих задачах.
Если вы уже пробовали работать с LLM, но хотите перейти от промптов и демо к более инженерному пониманию моделей, начните с бесплатного открытого урока:
20 мая в 20:00. «Основы парсинга сайтов».
Разберём, как с помощью Python скачивать информацию с сайтов, извлекать нужные данные и собирать основу для собственного датасета, который затем можно использовать в NLP‑задачах.
Полный список бесплатных уроков маясмотрите в дайджесте.
-Источник
 
Loading...
Error