Оптимизация SQL-запросов 1С-Битрикс

Наша компания занимается разработкой, поддержкой и обслуживанием решений на Битрикс и Битрикс24 любой сложности. От простых одностраничных сайтов до сложных интернет магазинов, CRM систем с интеграцией 1С и телефонии. Опыт разработчиков подтвержден сертификатами от вендора.
Предлагаемые услуги
Показано 1 из 1 услугВсе 1626 услуг
Оптимизация SQL-запросов 1С-Битрикс
Средняя
~1-2 недели
Часто задаваемые вопросы
Наши компетенции:
Этапы разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1165
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Разработка веб-сайта для компании ФИКСПЕР
    811
  • image_bitrix-bitrix-24-1c_development_of_an_online_appointment_booking_widget_for_a_medical_center_594_0.webp
    Разработка на базе Битрикс, Битрикс24, 1С для компании Development of an Online Appointment Booking Widget for a Medical Center
    563
  • image_bitrix-bitrix-24-1c_mirsanbel_458_0.webp
    Разработка на базе 1С Предприятие для компании МИРСАНБЕЛ
    743
  • image_crm_dolbimby_434_0.webp
    Разработка сайта на CRM Битрикс24 для компании DOLBIMBY
    655
  • image_crm_technotorgcomplex_453_0.webp
    Разработка на базе Битрикс24 для компании ТЕХНОТОРГКОМПЛЕКС
    976

Оптимизация SQL-запросов 1С-Битрикс

На нагруженном сайте Битрикс генерирует от 200 до 1000 SQL-запросов на страницу. Большинство из них — повторяющиеся, избыточные или с SELECT * вместо конкретных полей. Прежде чем покупать более дорогой сервер, есть смысл разобраться: какие именно запросы тормозят и почему.

Как найти проблемные запросы

Первый шаг — включить SQL-трекер Битрикса. В панели производительности (/bitrix/admin/perfmon_panel.php) переключите в режим профилирования. Либо программно:

\Bitrix\Main\Diag\SqlTracker::getInstance()->start();
// ... ваш код ...
$tracker = \Bitrix\Main\Diag\SqlTracker::getInstance();
$tracker->stop();
foreach ($tracker->getQueries() as $query) {
    echo $query->getSql() . ' — ' . $query->getTime() . 'ms' . PHP_EOL;
}

Трекер покажет все запросы за запрос, их время и стек вызовов. Ищите:

  • запросы дольше 50 мс — они тормозят страницу заметно
  • дубли — один и тот же запрос по 10–50 раз на страницу
  • запросы без WHERE по большим таблицам (b_iblock_element, b_sale_order_props)

Второй источник — slow_query_log MySQL/MariaDB. Включается через long_query_time = 0.5 в my.cnf. Даёт реальную картину под нагрузкой, не только при ручном тестировании.

Основные классы проблем

Запросы без индексов. b_iblock_element содержит от 10 000 до 1 000 000 строк. Запрос SELECT * FROM b_iblock_element WHERE IBLOCK_ID = 5 AND ACTIVE = 'Y' ORDER BY SORT без индекса по (IBLOCK_ID, ACTIVE, SORT) делает full table scan. Проверка: EXPLAIN SELECT ... — если в type видите ALL, индекса нет.

Битрикс создаёт индексы при установке, но не при добавлении пользовательских полей через b_uts_iblock_N_single (UTS-таблицы для пользовательских типов). Индексы по этим таблицам нужно добавлять вручную.

Избыточная выборка полей. CIBlockElement::GetList() по умолчанию джойнит несколько таблиц и возвращает десятки полей, включая DETAIL_TEXT весом в мегабайты. Если на странице нужны только ID, NAME и PREVIEW_PICTURE — задавайте $select явно:

CIBlockElement::GetList(
    ['SORT' => 'ASC'],
    ['IBLOCK_ID' => 5, 'ACTIVE' => 'Y'],
    false,
    ['nPageSize' => 20],
    ['ID', 'NAME', 'PREVIEW_PICTURE']  // только нужные поля
);

Без $select запрос включает LEFT JOIN b_iblock_element_iprop и другие таблицы, которые не нужны.

N+1 проблема. Классика: загрузили 20 элементов, потом в цикле для каждого отдельным запросом загружаете свойства. Итог — 21 запрос вместо 1–2. В старом API решается через $arSelectFields с включением свойств. В D7 — через fetchCollection() с fill() или предварительную загрузку через select().

Повторные запросы одних и тех же данных. Настройки сайта, группы пользователей, разделы инфоблоков — запрашиваются на каждой странице повторно. Стандартный кеш Битрикса (BXCache) должен это закрывать, но если кеш выключен или тегированный кеш сбрасывается слишком часто — запросы идут в БД.

Работа с индексами

Добавление недостающих индексов — самое быстрое решение с наибольшим эффектом:

Таблица Рекомендуемый индекс Когда нужен
b_iblock_element (IBLOCK_ID, ACTIVE, SORT) Почти всегда
b_iblock_element_property (IBLOCK_PROPERTY_ID, VALUE) При фильтрации по свойствам
b_sale_order (USER_ID, STATUS_ID, DATE_INSERT) Личный кабинет покупателя
b_sale_basket (ORDER_ID, FUSER_ID) Корзина, оформление заказа
b_search_content_stem (PARAM2) Поиск по большому каталогу

Индекс добавляется через SQL или Битрикс ORM:

$connection = \Bitrix\Main\Application::getConnection();
$connection->queryExecute(
    "ALTER TABLE b_iblock_element_property ADD INDEX ix_prop_val (IBLOCK_PROPERTY_ID, VALUE(64))"
);

Осторожно с VALUE(64): строковые поля индексируются по префиксу. Для числовых значений, хранящихся как VARCHAR, рассмотрите виртуальный столбец с приведением типа.

Оптимизация через D7 ORM

D7 ORM даёт возможность точечно управлять запросом. Сравните:

// Плохо: загружает все поля, все свойства
$result = \Bitrix\Iblock\ElementTable::getList([
    'filter' => ['IBLOCK_ID' => 5, 'ACTIVE' => 'Y'],
]);

// Лучше: только нужные поля, явный лимит
$result = \Bitrix\Iblock\ElementTable::getList([
    'select' => ['ID', 'NAME', 'PREVIEW_PICTURE_ID'],
    'filter' => ['IBLOCK_ID' => 5, 'ACTIVE' => 'Y'],
    'order'  => ['SORT' => 'ASC'],
    'limit'  => 20,
    'offset' => 0,
    'cache'  => ['ttl' => 3600],
]);

Параметр cache в запросе ORM — встроенное кеширование на уровне запроса. Инвалидация — через тегированный кеш при изменении элемента инфоблока.

Кеш как замена лишним запросам

Если данные меняются раз в час — нет смысла ходить в БД на каждый запрос. Используйте \Bitrix\Main\Data\Cache:

$cache = \Bitrix\Main\Data\Cache::createInstance();
$cacheId = 'catalog_top_' . $iblockId;
$cachePath = '/catalog/top/';

if ($cache->initCache(3600, $cacheId, $cachePath)) {
    $data = $cache->getVars();
} elseif ($cache->startDataCache()) {
    $tagCache = new \Bitrix\Main\Data\TaggedCache();
    $tagCache->startTagCache($cachePath);

    $data = /* ваш запрос */;

    $tagCache->registerTag('iblock_id_' . $iblockId);
    $tagCache->endTagCache();
    $cache->endDataCache($data);
}

При изменении любого элемента инфоблока тег iblock_id_N инвалидируется автоматически, и следующий запрос идёт в БД.

Сроки работ

Задача Объём работ Ожидаемый эффект
Профилирование, выявление топ-10 запросов 1 день Понимание проблемы
Добавление недостающих индексов 1–2 дня Ускорение в 2–10 раз на конкретных запросах
Оптимизация $select в компонентах 2–3 дня Снижение нагрузки на 20–40%
Устранение N+1, добавление кеша 3–5 дней Снижение числа запросов в 3–5 раз
Комплекс: индексы + выборка + кеш + ORM 1–2 недели Страница с 500+ запросов → 50–100 запросов

Оптимизация SQL — не разовое действие. После изменений нужен регулярный мониторинг: slow query log раз в неделю, трекер при релизах новых компонентов. Проблемы накапливаются по мере роста каталога и объёма данных.