Форум программистов, компьютерный форум, киберфорум
PHP: ООП
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
 
Рейтинг 4.77/26: Рейтинг темы: голосов - 26, средняя оценка - 4.77
Заблокирован

Суть DI в чём?

05.06.2020, 12:25. Показов 5617. Ответов 43
Метки php (Все метки)

Студворк — интернет-сервис помощи студентам
index.php
PHP
1
2
<?php
require_once 'engine/bootstrap.php';
service.php
PHP
1
2
3
4
<?php
return [
    \engine\service\router\Provider::class
];
bootstrap.php
PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<?php
require_once dirname(__DIR__) . '/vendor/autoload.php';
 
use engine\Cms;
use engine\di\Di;
 
try {
    $di = new Di();
    $services = require_once 'config/service.php';
    foreach ($services as $service) {
        $provider = new $service($di);
        $provider->init();
    }
    $cms = new Cms($di);
    $cms->run();
} catch (Error $error) {
    echo $error->getMessage();
}
Теперь вопросы:

1. Зачем вся эта муть, почему нельзя сделать так:
index.php
PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<?php
// require_once dirname(__DIR__) . '/vendor/autoload.php'; // удалено - ЗАЧЕМ ЭТО ТУТ!!!???
 
use engine\Cms;
use engine\di\Di;
 
try {
    $di = new Di(); // внутри заполняем container
    $services = require_once 'config/service.php';
    foreach ($services as $service) {
        $provider = new $service($di); // СРАЗУ СОЗДАЁМ ОБЪЕКТ(Ы) И ПИХАЕМ В ТОЧКУ ДОСТУПА!!!???
        $provider->init();
    }
    $cms = new Cms($di);
    $cms->run();
} catch (Error $error) {
    echo $error->getMessage();
}
И всё.

2. Я правильно понимаю, что у меня в $di->container будет массив СРАЗУ ВСЕХ объектов приложения, то есть они СРАЗУ УЖЕ СОЗДАНЫ, зачем мне куча УЖЕ СОЗДАННЫХ экземпляров, а не создание по мере обращения? Или я что-то не понимаю - это же так называемый РЕЕСТР.
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
05.06.2020, 12:25
Ответы с готовыми решениями:

В чем суть интерфейсов?
За день я в интернете начиталась столько всего про интерфейсы, что запуталась до нельзя!!! И так, звучит главный вопрос: в чем же суть...

В чем суть PHP?
Ребят, подскажите пожалуйста правильно ли я понял суть PHP. Вот лежит на сервере код определенный. Мы обращаемся по домену и этот код...

В чем суть Invalid Parameter
Invalid Parameter – yii\base\InvalidParamException Hash is invalid. Добавлено через 27 минут вот код куда ругается in...

43
Заблокирован
11.06.2020, 17:23  [ТС]
Студворк — интернет-сервис помощи студентам
sad67man,

PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
$di = new Di();
$services = require_once 'config/service.php'; // массив неймспейсов ПРОВАЙДЕРОВ сервисов
foreach ($services as $service) {
    $provider = new $service($di);
}
 
class Di
{
    private $container;
    
    public function set($key, $value)
    {
        $this->container[$key] = $value;
    }
 
    public function get($key)
    {
        return new $this->container[$key];
    }
}
 
// в каждом провайдере конструктор с кодом $di->set('имя_сервиса', неймспейс_СЕРВИСА);
я правильно суть понял?
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 18:33
Лучший ответ Сообщение было отмечено Verolomstvo как решение

Решение

Verolomstvo, Не совсем

должно быть

$di->set("нейспейс_Сервиса", функция для создания объекта.);

Данную функцию можно поместить внутрь какого-нибудь класса - факторки

$di->setFactory("нейспейс_Сервиса", 'неймспейс_класса_факторки');

Вообще по PSR у контейнера главным образом должны быть методы get() и has(), как вы его уже будете заполнять сами решаете.

Объект должен создаваться в момент вызова $di->get("нейспейс_Сервиса");
При повторном вызове $di->get("нейспейс_Сервиса") должен браться уже созданный объект.

Причем мы можем обойтись и без метода set;
При вызове $di->get("нейспейс_Сервиса"). Di контейнер должен автоматически попытаться создать класс, при этом парсить его конструктор через рефлексию на типы принимаемых данных, и рекурсивно создавать все его зависимости через $di->get().

Так же можно сопоставлять интерфейсы
$di->set('интерфейс_Серсиса', нейспейс_Сервиса/фнкция/факторка);
Теперь если какой-либо класс будет зависеть не от конкретного объекта, а от интерфейса, DI контейнер будет подставлять данный класс.

Вы конечно можете сами такое реализовать. Но есть и готовые решения к примеру https://php-di.org/doc/getting-started.html
1
Заблокирован
11.06.2020, 18:39  [ТС]
sad67man, я считаю, что не нужна функция создания, так как в get по имени сервиса и его неймспейсу можно тупо сделать new, ну и завернуть в синглтон по ходу, а то я уже забыл про этот нюанс.
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 18:48
Verolomstvo, Через контейнер подлючаются классы - службы. Которые могут иметь настройки. К примеру

PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
interface Renderer
{
    public function render(string $name, array $params = []): string
}
 
class PhpRenderer implements Renderer
{
    private $templatesPath;
 
    public function __construct($templatesPath)
    {
        $this->templatesPath = $templatesPath;
    }
 
    public function render(string $name, array $params = []): string
    {
        ob_start();
        extract($params);
        require $this->templatesPath . '/' . $name . '.php';
        return ob_get_clean();
    }
}
 
class MyController
{
    private $renderer;
 
    public function __construct(Renderer $renderer)
    {
        $this->renderer = $renderer;
    }
 
    public function actionIndex() 
    {
        return $this->renderer->render('index');
    }
 
}
 
$container->set(Renderer::class, function() {
    return new PhpRenderer(ROOT_DIR . '/views');
})
 
$controller = $container->get(MyController::class);
Добавлено через 3 минуты
Потом допустим вы захотите поменять PhpRenderer на Twig. Вам необходимо будет написать свой класс TwigRenderer, который будет реализовывать интерфейс Renderer. И заменить в контейнере зависимостей.
0
Заблокирован
11.06.2020, 18:54  [ТС]
sad67man,

провайдер для Render

PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class Provider extends AbstractProvider
{
    public function __construct()
    {
        $this->di->set('render', неймспейс_класса_Render);
    }
}
 
$di = new Di();
$services = require_once 'config/service.php'; // массив неймспейсов ПРОВАЙДЕРОВ сервисов
foreach ($services as $service) {
    $provider = new $service($di);
}
 
// массив $di->container на данный момент имеет вид ['render' => неймспейс_класса_Render]
// так как $this->di у нас пронизывает весь скрипт, то в любом месте делаем
$this->di->get('render'); // отработает return new $this->container[$key]
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 18:56
Цитата Сообщение от Verolomstvo Посмотреть сообщение
ну и завернуть в синглтон по ходу
Di контейнер как раз заменяет антипатерн Синглтон. А вы хотите его использовать Синглтон для реализации контейнера?
0
Заблокирован
11.06.2020, 18:58  [ТС]
Цитата Сообщение от sad67man Посмотреть сообщение
Объект должен создаваться в момент вызова $di->get("нейспейс_Сервиса");
При повторном вызове $di->get("нейспейс_Сервиса") должен браться уже созданный объект.
Цитата Сообщение от sad67man Посмотреть сообщение
Di контейнер как раз заменяет антипатерн Синглтон. А вы хотите его использовать Синглтон для реализации контейнера?
ну я подумал что тут на синглтон намёк идёт
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 18:59
Verolomstvo, Неймспейс класса - это же строка. Он и сам может являться ключем для получения объекта из контейнера.
0
Заблокирован
11.06.2020, 19:02  [ТС]
sad67man, мне не нужна строкаааааааааааааа, мне нужен удобночитаемый ключ + значение в виде строкииииииииии неймспейса
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 19:07
Verolomstvo, вы не совсем понимаете суть DI контейнера

Допустим у нас есть такой класс

PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
class InvoiceInService
{
    private $suppliers;
    private $creator;
    private $invoices;
    private $api;
 
    public function __construct(
        SupplierRepository $suppliers,
        InvoiceInCreator $creator,
        InvoiceInRepository $invoices,
        MoySkladAPI $api
    )
    {
        $this->suppliers = $suppliers;
        $this->creator = $creator;
        $this->invoices = $invoices;
        $this->api = $api;
    }
 
    public function create(InvoiceInForm $form): InvoiceIn
    {
        $supplier = $this->suppliers->getById($form->supplier_id);
        $invoice = $this->creator->create($supplier, $form->getFilePath());
        $this->invoices->save($invoice);
        return $invoice;
    }
 
    public function delete($id)
    {
        $invoice = $this->invoices->getById($id);
        $this->invoices->delete($invoice);
    }
 
    public function sendToMoySklad(int $id): void
    {
        $invoice = $this->invoices->getById($id);
        $moysklad_id = $this->api->createInvoiceIn($invoice);
        $invoice->sent($moysklad_id);
        $this->invoices->save($invoice);
    }
}
Это класс-служба. И все что подключается через конструктор - являются его зависимостями. Di контейнер так и называется Контейнер внедрения зависимостей. Т.е. его задача автоматически подставлять классы в конструкторы. А не дергать его через $this->di->get()

Если это вам не нужно, это другой вопрос, только тогда это будет не DI контейнер, а что-то уже другое.
0
Заблокирован
11.06.2020, 19:13  [ТС]
Цитата Сообщение от sad67man Посмотреть сообщение
А не дергать его через $this->di->get()
ну так бля я с этого и начинал: объекты УЖЕ СОЗДАНЫ ВСЕ И ВСЕГДА НАХОДЯТСЯ В МАССИВЕ и передаются готовые?

или всё-таки их ещё нужно вызвать по ключу массива МЕТОДОМ МАТЬ ЕГО get('ключ_вызываемого_класса') класса Di где мы его до этого заполнили провайдерами методом set?
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 19:56
Лучший ответ Сообщение было отмечено Verolomstvo как решение

Решение

Verolomstvo, Объекты создаются в момент вызова через get(),
Но создаваемый класс может принимать другие классы в конструктор - зависимости. Получается, чтоб его создать, нужно создать сначала те классы, которые он ожидает в конструкторе. А они в свою очередь могут тоже иметь параметры в конструкторе. Т.е. чтоб их создать нужно создать и зависимости для них.

Т.е. главным образом вы должны создать через метод get() сам контроллер. А все остальное что нужно для его создание Di контейнер в этот момент создаст. Но только то, что необходимо. Т.е. если вы нигде во внутренних классах не используете объект подключения к БД, То Di контейнер не будет создавать этот объект. А если у вас несколько внутренних классов требует подключение к БД. То Контейнер создаст один объект подключения к БД и прокинет его в конструкторы данных классов.

По факту объекты создаются в момент вызова get(), но не по одному - а пачками, включая все зависимости создаваемого объекта.
1
Заблокирован
11.06.2020, 20:14  [ТС]
sad67man,ну так я же в get делаю new, берём в любом классе DI и вызываем нужный класс методом get, в методе get ищем по ключу неймспейс нужного класса из массива (пачки) неймспейсов, созданного провайдерами
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 20:56
Цитата Сообщение от Verolomstvo Посмотреть сообщение
берём в любом классе DI и вызываем нужный класс методом get,
Получается, что все классы принимают в конструтор только один объект контейнера? А зачем? Чем это будет отличаться, если вы просто напишите функцию или статический метод. Это уже будет не ООП, В любом случае IDE-редактор станет бесполезным.
1
Заблокирован
11.06.2020, 21:17  [ТС]
sad67man, я вообще не знаю зачем я в это полез, мне это, как я понимаю, не нужно

время только трачу, зачем? мне это не подходит, я прекрасно обращаюсь с объектами по классике даже
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
11.06.2020, 21:37
Verolomstvo, Контейнер зависимости облегчает жизнь. Только нужно его правильно использовать. А так конечно ваше дело)
1
Заблокирован
11.06.2020, 22:35  [ТС]
sad67man, ладно хоть поверхностно ознакомился, для общего так сказать, развития, спс

Добавлено через 41 минуту
Цитата Сообщение от sad67man Посмотреть сообщение
Получается, что все классы принимают в конструтор только один объект контейнера?
PHP
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public function dispatch(Di $di)
{
    if ($this->matchRoute()) {
        $controller = 'app\controller\\' . $this->route['controller'] . 'Controller';
        if (class_exists($controller)) {
            $cObj = new $controller($di);
            $action = $this->route['action'] . 'Action';
            if (method_exists($cObj, $action)) {
                $cObj->$action($this->route);
            } else {
                throw new Error("Метод {$controller}::{$action} не найден", 404);
            }
        } else {
            throw new Error("Контроллер {$controller} не найден", 404);
        }
    } else {
        throw new Error('Страница не найдена', 404);
    }
}
все классы принимают в себя класс Di с МАССИВОМ $di->container [ключ => неймспейс_класса] из которого можно в любом месте запустить любой класс(Ы) методом get по ключу
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
12.06.2020, 00:21
Цитата Сообщение от Verolomstvo Посмотреть сообщение
се классы принимают в себя класс Di с МАССИВОМ $di->container [ключ => неймспейс_класса] из которого можно в любом месте запустить любой класс(Ы) методом get по ключу
И чем от отличается, если бы вы просто сделали статический метод Di::get() и дергали в любом месте? Почему бы не подключить нормальный Di контейнер через composer и не создавать через него сам контроллер?
PHP
1
$cObj = $di->get($controller);
Добавлено через 2 минуты
Или вообще создать через него само приложение

PHP
1
2
3
//index.php
$conteiner = require dirname(__DIR__) . '/config/container.php';
$app = $container->get(Application::class);
1
Заблокирован
12.06.2020, 00:42  [ТС]
sad67man, это то?
Вложения
Тип файла: zip DI-test.zip (1.17 Мб, 6 просмотров)
0
 Аватар для sad67man
2603 / 1507 / 689
Регистрация: 23.08.2015
Сообщений: 3,814
12.06.2020, 00:50
Verolomstvo, Нет..
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
12.06.2020, 00:50
Помогаю со студенческими работами здесь

В чём суть ошибки? Warning: Parameter 3 to showBlogSection() expected to be a reference, value given in on line 100
Здравствуйте. Не могу понять в чем суть ошибки Warning: Parameter 3 to showBlogSection() expected to be a reference, value given in on line...

В чём суть сборки?
Здравствуйте. Я верю, что тут действительно есть знатоки компьютерного железа и я обращаюсь к вам. Я сам как бы начинающий IT-шник и учусь...

В чем суть continue в if-else
код первый. прата глава 7 упражнение 3. оператор continue отсутствует, все прекрасно работает. счетчик вынесен в отдельный оператор ?: 1...

В чем суть операторов << и >>
Здравствуйте. Уважаемые Форумчане, нужна помощь. Имеется следующий код if ((PINB&amp;(1 &lt;&lt; PB0)) == 0) // Если на выводе...

Интерфейсы - в чем их суть
В чем суть интерфейсов объясните пожалуйста. Добавлено через 19 минут А если быть точнее, то какова их практическая полезность?...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
40
Ответ Создать тему
Новые блоги и статьи
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост.
Programma_Boinc 28.12.2025
Советы по крайней бережливости. Внимание, это ОЧЕНЬ длинный пост. Налог на собак: https:/ / **********/ gallery/ V06K53e Финансовый отчет в Excel: https:/ / **********/ gallery/ bKBkQFf Пост отсюда. . .
Кто-нибудь знает, где можно бесплатно получить настольный компьютер или ноутбук? США.
Programma_Boinc 26.12.2025
Нашел на реддите интересную статью под названием Anyone know where to get a free Desktop or Laptop? Ниже её машинный перевод. После долгих разбирательств я наконец-то вернула себе. . .
Thinkpad X220 Tablet — это лучший бюджетный ноутбук для учёбы, точка.
Programma_Boinc 23.12.2025
Рецензия / Мнение/ Перевод Нашел на реддите интересную статью под названием The Thinkpad X220 Tablet is the best budget school laptop period . Ниже её машинный перевод. Thinkpad X220 Tablet —. . .
PhpStorm 2025.3: WSL Terminal всегда стартует в ~
and_y87 14.12.2025
PhpStorm 2025. 3: WSL Terminal всегда стартует в ~ (home), игнорируя директорию проекта Симптом: После обновления до PhpStorm 2025. 3 встроенный терминал WSL открывается в домашней директории. . .
Как объединить две одинаковые БД Access с разными данными
VikBal 11.12.2025
Помогите пожалуйста !! Как объединить 2 одинаковые БД Access с разными данными.
Новый ноутбук
volvo 07.12.2025
Всем привет. По скидке в "черную пятницу" взял себе новый ноутбук Lenovo ThinkBook 16 G7 на Амазоне: Ryzen 5 7533HS 64 Gb DDR5 1Tb NVMe 16" Full HD Display Win11 Pro
Музыка, написанная Искусственным Интеллектом
volvo 04.12.2025
Всем привет. Некоторое время назад меня заинтересовало, что уже умеет ИИ в плане написания музыки для песен, и, собственно, исполнения этих самых песен. Стихов у нас много, уже вышли 4 книги, еще 3. . .
От async/await к виртуальным потокам в Python
IndentationError 23.11.2025
Армин Ронахер поставил под сомнение async/ await. Создатель Flask заявляет: цветные функции - провал, виртуальные потоки - решение. Не threading-динозавры, а новое поколение лёгких потоков. Откат?. . .
Поиск "дружественных имён" СОМ портов
Argus19 22.11.2025
Поиск "дружественных имён" СОМ портов На странице: https:/ / norseev. ru/ 2018/ 01/ 04/ comportlist_windows/ нашёл схожую тему. Там приведён код на С++, который показывает только имена СОМ портов, типа,. . .
Сколько Государство потратило денег на меня, обеспечивая инсулином.
Programma_Boinc 20.11.2025
Сколько Государство потратило денег на меня, обеспечивая инсулином. Вот решила сделать интересный приблизительный подсчет, сколько государство потратило на меня денег на покупку инсулинов. . . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2025, CyberForum.ru