Налаштування DI-контейнера
Огляд варіантів конфігурації контейнера Nette DI.
Конфігураційний файл
Контейнером Nette DI легко керувати за допомогою конфігураційних файлів. Зазвичай вони записуються у форматі NEON format. Ми рекомендуємо використовувати для редагування таких файлів редактори з підтримкою цього формату.
decorator: Decorator
di: DI Container
extensions: Встановіть додаткові розширення DI
includes: Включення файлів
parameters: Параметри
search: Автоматична реєстрація сервісів
services: Сервіси
Щоб записати рядок, що містить символ %
, вы должны
экранировать его удвоением до %%
.
Параметри
Можна задати параметри, які потім можуть бути використані як частина визначення сервісу. Це може допомогти розділити значення, які ви хочете змінювати більш регулярно:
parameters:
dsn: 'mysql:host=127.0.0.1;dbname=test'
user: root
password: secret
У будь-якому файлі конфігурації можна посилатися на параметр
foo
через %foo%
в іншому місці. Вони також можуть
використовуватися всередині рядків, таких як '%wwwDir%/images'
.
Параметри не повинні бути тільки рядками, вони можуть бути також значеннями масиву:
parameters:
mailer:
host: smtp.example.com
secure: ssl
user: franta@gmail.com
languages: [cs, en, de]
Можна посилатися на окремий ключ так: %mailer.user%
.
Якщо вам потрібно отримати значення будь-якого параметра у вашому коді, наприклад, у класі, то передайте його цьому класу. Наприклад, у конструкторі. Не існує глобального об'єкта конфігурації, який міг би запитувати значення параметрів. Це суперечить принципу впровадження залежностей.
Сервіси
Див. окремий розділ.
Декоратор
Як масово редагувати всі сервіси певного типу? Потрібно викликати певний метод для всіх презентерів, що успадковують від певного спільного предка? Ось звідки береться декоратор:
decorator:
# Для всіх сервісів, які є екземплярами цього класу або інтерфейсу
App\UI\BasePresenter:
setup:
- setProjectId(10) # викликаємо цей метод
- $absoluteUrls = true # і задаємо змінну
Декоратор також можна використовувати для встановлення тегів або ввімкнення режиму впровадження.
decorator:
InjectableInterface:
tags: [mytag: 1]
inject: true
DI
Технічні налаштування контейнера DI:
di:
# відображати DIC в панелі налагодження Tracy?
debugger: ... # (bool) за замовчуванням true
# типи параметрів, які не потрібно монтувати автоматично
excluded: ... # (string[])
# увімкнути створення лінивих сервісів?
lazy: ... # (bool) за замовчуванням false
# клас, від якого успадковується контейнер DI
parentClass: ... # (string) за замовчуванням Nette\DI\Container
Ліниві сервіси
Параметр lazy: true
дозволяє ліниве (відкладене) створення
сервісів. Це означає, що сервіси створюються не тоді, коли вони
запитуються з контейнера DI, а лише після їх першого використання. Це
може пришвидшити запуск програми та зменшити використання пам'яті,
оскільки створюються лише ті служби, які потрібні для конкретного
запиту.
Для конкретного сервісу ліниве створення можна налаштувати.
Ліниві об'єкти можна використовувати лише для визначених користувачем класів, а не для внутрішніх класів PHP. Потрібна версія PHP 8.4 або новіша.
Експорт метаданих
Клас контейнера DI також містить безліч метаданих. Ви можете зменшити його, скоротивши експорт метаданих.
di:
export:
# експортувати параметри
parameters: false # (bool) за замовчуванням true
# експортувати теги
tags: # (string[]|bool) за замовчуванням усі
- event.subscriber
# експортувати дані для автоматичного підключення
types: # (string[]|bool) за замовчуванням усі
- Nette\Database\Connection
- Symfony\Component\Console\Application
Якщо ви не використовуєте масив $container->getParameters()
, можна
вимкнути експорт параметрів. Крім того, ви можете експортувати тільки
ті теги, через які ви отримуєте сервіси, використовуючи метод
$container->findByTag(...)
. Якщо ви не викликаєте цей метод зовсім, можна
повністю відключити експорт тегів, вказавши значення false
.
Ви можете значно зменшити метадані для автоматичного підключення,
вказавши класи, які ви використовуєте як параметр у методі
$container->getByType()
. І знову, якщо ви не викликаєте цей метод зовсім
(або тільки в bootstrap для отримання
Nette\Application\Application
), можна повністю відключити експорт, вказавши
значення false
.
Розширення
Реєстрація інших розширень DI. Таким чином, ми додаємо, наприклад, DI
розширення Dibi\Bridges\Nette\DibiExtension22
під іменем dibi
:
extensions:
dibi: Dibi\Bridges\Nette\DibiExtension22
Потім ми налаштовуємо його в секції, яка також називається
dibi
:
dibi:
host: localhost
Ви також можете додати клас розширення з параметрами:
extensions:
application: Nette\Bridges\ApplicationDI\ApplicationExtension(%debugMode%, %appDir%, %tempDir%/cache)
Файли, що включаються
Додаткові файли конфігурації можуть бути вставлені в секції
includes
:
includes:
- parameters.php
- services.neon
- presenters.neon
Назва parameters.php
не є друкарською помилкою, конфігурацію також
можна записати в PHP-файлі, який повертає її у вигляді масиву:
<?php
return [
'database' => [
'main' => [
'dsn' => 'sqlite::memory:',
],
],
];
Якщо в конфігураційних файлах з'являються елементи з однаковими
ключами, вони будуть перезаписані або об'єднані у
випадку з масивами. Наступний включений файл має вищий пріоритет, ніж
попередній. Файл, у якому вказано секцію includes
, має вищий
пріоритет, ніж файли, включені в нього.
Пошук
Автоматичне додавання сервісів у контейнер DI робить роботу надзвичайно приємною. Nette автоматично додає презентери в контейнер, але ви можете легко додати будь-які інші класи.
Просто вкажіть, у яких каталогах (і підкаталогах) слід шукати класи:
search:
- in: %appDir%/Forms
- in: %appDir%/Model
Зазвичай, однак, ми не хочемо додавати всі класи та інтерфейси, тому ми можемо відфільтрувати їх:
search:
- in: %appDir%/Forms
# фільтрація за ім'ям файлу (string|string[])
files:
- *Factory.php
# фільтрація за ім'ям класу (string|string[])
classes:
- *Factory
Або ми можемо вибрати класи, які успадковують або реалізують принаймні один із наступних класів:
search:
- in: %appDir%
extends:
- App\*Form
implements:
- App\*FormInterface
Ви також можете визначити негативні правила, тобто маски імен класів або предків, і якщо вони відповідають вимогам, сервіс не буде додано в контейнер DI:
search:
- in: %appDir%
exclude:
файли: ...
classes: ...
extends: ...
implements: ...
Для додаткових сервісів можна визначити теги:
search:
- in: %appDir%
tags: ...
Об'єднання
Якщо елементи з однаковими ключами з'являються в кількох конфігураційних файлах, їх буде перезаписано або об'єднано в разі масивів. Пізніший включений файл має вищий пріоритет.
config1.neon | config2.neon | результат |
---|---|---|
|
|
|
Щоб запобігти об'єднанню певного масиву, використовуйте знак оклику відразу після імені масиву:
config1.neon | config2.neon | result |
---|---|---|
|
|
|