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

Наша компанія займається розробкою, підтримкою та обслуговуванням рішень на Бітрікс та Бітрікс24 будь-якої складності. Від простих односторінкових сайтів до складних інтернет-магазинів, CRM систем з інтеграцією 1С та телефонії. Досвід розробників підтверджено сертифікатами від вендора.
Пропоновані послуги
Показано 1 з 1 послугУсі 1626 послуг
Оптимізація SQL-запитів 1С-Бітрікс
Середня
~1-2 тижні
Часті питання

Наші компетенції:

Етапи розробки

Останні роботи

  • image_website-b2b-advance_0.png
    Розробка сайту компанії B2B ADVANCE
    1262
  • image_bitrix-bitrix-24-1c_fixper_448_0.png
    Розробка веб-сайту для компанії ФІКСПЕР
    851
  • 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
    585
  • image_bitrix-bitrix-24-1c_mirsanbel_458_0.webp
    Розробка на базі 1С Підприємство для компанії МИРСАНБЕЛ
    751
  • image_crm_dolbimby_434_0.webp
    Розробка сайту на CRM Бітрікс24 для компанії DOLBIMBY
    657
  • image_crm_technotorgcomplex_453_0.webp
    Розробка на базі Бітрікс24 для компанії ТЕХНОТОРГКОМПЛЕКС
    989

Оптимізація 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 раз на тиждень, трекер при релізах нових компонентів. Проблеми накопичуються по мірі зростання каталогу та обсягу даних.