Как я могу использовать контейнер IOC Laravel вне Laravel для инъекции метода

Короткий рассказ:
Я не могу получить инъекцию метода, работающую с контейнером Laravel, установленным с помощью composer (https://packagist.org/packages/illuminate/container). Инъекция работает только при использовании в конструкторе объектов. Например:

class SomeClass {
function __construct(InjectedClassWorksHere $obj) {}
function someFunction(InjectedClassFailsHere $obj) {}
}

Длинная история:
Я думал о переосмыслении крупного проекта с использованием Laravel, но из-за давления со стороны бизнеса я не могу инвестировать в то время, которое мне хотелось бы. Чтобы не выбрасывать «ребенка с водой из ванны», я использую отдельные компоненты Laravel для повышения элегантности кода, разрабатываемого в старой ветке. Одним из моих любимых новых методов, которые я выбрал при оценке Laravel, была концепция внедрения зависимостей. Позже я был рад узнать, что могу использовать это ВНЕ проекта Laravel. Теперь у меня это работает, и все хорошо, за исключением того, что dev-версия контейнера, найденная в сети, похоже, не поддерживает внедрение метода.

Кто-нибудь еще мог заставить контейнер работать и делать инъекцию методов вне проекта Laravel?

Мой подход пока …

composer.json

"illuminate/support": "5.0.*@dev",
"illuminate/container": "5.0.*@dev",

Загрузочный код приложения:

use Illuminate\Container\Container;

$container = new Container();
$container->bind('app', self::$container); //not sure if this is necessary

$dispatcher = $container->make('MyCustomDispatcher');
$dispatcher->call('some URL params to find controller');

С учетом вышесказанного я могу вводить в конструкторы моих контроллеров, но не методы их методов. Что мне не хватает?

Полный исходный код … (C: \ workspace \ LMS> php cmd \ test_container.php)

<?php

// This sets up my include path and calls the composer autoloader
require_once "bare_init.php";

use Illuminate\Container\Container;
use Illuminate\Support\ClassLoader;
use Illuminate\Support\Facades\Facade;

// Get a reference to the root of the includes directory
$basePath = dirname(dirname(__FILE__));

ClassLoader::register();
ClassLoader::addDirectories([
$basePath
]);

$container = new Container();
$container->bind('app', $container);
$container->bind('path.base', $basePath);

class One {
public $two;
public $say = 'hi';
function __construct(Two $two) {
$this->two = $two;
}
}

Class Two {
public $some = 'thing';
public function doStuff(One $one) {
return $one->say;
}
}

/* @var $one One */
$one = $container->make(One);
var_dump($one);
print $one->two->doStuff();

Когда я запускаю вышеупомянутое, я получаю …

C:\workspace\LMS>php cmd\test_container.php
object(One)#9 (2) {
["two"]=>
object(Two)#11 (1) {
["some"]=>
string(5) "thing"}
["say"]=>
string(2) "hi"}

PHP Catchable fatal error:  Argument 1 passed to Two::doStuff() must be an instance of One, none
given, called in C:\workspace\LMS\cmd\test_container.php on line 41
and defined in C:\workspace\LMS\cmd\test_container.php on line 33

Catchable fatal error: Argument 1 passed to Two::doStuff() must be an instance of One, none
given, called in C:\workspace\LMS\cmd\test_container.php on line 41 and
defined in C:\workspace\LMS\cmd\test_container.php on line 33

Или, более простой пример, который иллюстрирует внедрение, работающее в конструкторе, но не в методе …

class One {
function __construct(Two $two) {}
public function doStuff(Three $three) {}
}

class Two {}
class Three {}

$one = $container->make(One); // totally fine. Injection works
$one->doStuff(); // Throws Exception. (sad trombone)

4

Решение

Просто передайте пример One в ваш звонок Two:

$one = $container->make('One');
var_dump($one);
print $one->two->doStuff($one);

возвращается …

object(One)#8 (2) {
["two"]=>
object(Two)#10 (1) {
["some"]=>
string(5) "thing"}
["say"]=>
string(2) "hi"}
hi

Обновление: исправленный ответ после дальнейших исследований.

Как упоминалось ниже, в Laravel 5.0 внедрение метода доступно только на маршрутах и ​​контроллерах. Таким образом, вы можете включить их в свой проект и получить немного больше Laravel-у в процессе. Вот как:

В composer.json, нужно добавить в illuminate/routing а также illuminate/events:

{
"require-dev": {
"illuminate/contracts": "5.0.*@dev",
"illuminate/support": "5.0.*@dev",
"illuminate/container": "5.0.*@dev",
"illuminate/routing": "5.0.*@dev",
"illuminate/events": "5.0.*@dev"},
"autoload": {
"psr-4": {
"App\\": "app/"}
},
"minimum-stability": "dev",
"prefer-stable": true
}

В routing.php, настроить службы маршрутизации и управления Laravel:

/**
* routing.php
*
* Sets up Laravel's routing and controllers
*
* adapted from http://www.gufran.me/post/laravel-components
* and http://www.gufran.me/post/laravel-illuminate-router-package-in-your-application
*/
$basePath = str_finish(dirname(__FILE__), '/app/');

$controllersDirectory = $basePath . 'Controllers';

// Register directories into the autoloader
Illuminate\Support\ClassLoader::register();
Illuminate\Support\ClassLoader::addDirectories($controllersDirectory);

// Instantiate the container
$app = new Illuminate\Container\Container();
$app['env'] = 'production';

$app->bind('app', $app); // optional
$app->bind('path.base', $basePath); // optional

// Register service providers
with (new Illuminate\Events\EventServiceProvider($app))->register();
with (new Illuminate\Routing\RoutingServiceProvider($app))->register();

require $basePath . 'routes.php';

$request = Illuminate\Http\Request::createFromGlobals();
$response = $app['router']->dispatch($request);
$response->send();

В Controllers/One.php, создайте класс в качестве контроллера, чтобы мы могли использовать инъекцию метода L5:

/**
* Controllers/One.php
*/
Class One extends Illuminate\Routing\Controller {
public $some = 'thingOne';
public $two;
public $three;

function __construct(Two $two) {
$this->two = $two;
echo('<pre>');
var_dump ($two);
echo ($two->doStuffWithTwo().'<br><br>');
}

public function doStuff(Three $three) {
var_dump ($three);
return ($three->doStuffWithThree());
}
}

В routes.php, Определите наш тестовый маршрут:

$app['router']->get('/', 'One@dostuff');

Наконец, в index.php, загрузите все и определите наши классы для проверки внедрения зависимости:

/**
* index.php
*/

// turn on error reporting
ini_set('display_errors',1);
error_reporting(E_ALL);

require 'vendor/autoload.php';
require 'routing.php';

// the classes we wish to inject
Class Two {
public $some = 'thing Two';
public function doStuffWithTwo() {
return ('Doing stuff with Two');
}
}
Class Three {
public $some = 'thing Three';
public function doStuffWithThree() {
return ('Doing stuff with Three');
}
}

Нажмите index.php, и вы должны получить это:

object(Two)#40 (1) {
["some"]=>
string(9) "thing Two"}
Doing stuff with Two

object(Three)#41 (1) {
["some"]=>
string(11) "thing Three"}
Doing stuff with Three

Некоторые заметки …

  • Нет необходимости явно связывать классы. Ларавел позаботится об этом.
  • Теперь у вас есть дополнительный бонус от маршрутизации и контроллеров Laravel
  • Это работает, потому что теперь нам не нужно звонить $one->doStuff();с пустым параметром, который выдает исключение (так как doStuff ожидает экземпляр). Вместо этого маршрутизатор вызывает doStuff и разрешает контейнер IoC для нас.
  • Кредит для http://www.gufran.me/post/laravel-illuminate-router-package-in-your-application который проведет вас через все это, и который, кажется, вдохновил на проект Мэтта Штауфера, упомянутый выше. Оба очень крутые и достойны прочтения.
6

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]