部分依赖注入

Partial dependency injection

在 MVC 上下文中,我有一个依赖于服务的控制器,该服务又依赖于 data_source(在特定情况下,客户端从第三方获取数据 API).

为了在测试时使用模拟 data_source 实例化服务,服务的构造函数需要一个 data_source。 这同样适用于控制器,其构造函数需要服务。

创建控制器时,我也想向它传递一个请求对象,因为我更喜欢这个

new Controller(request, service).action_name

至此

new Controller(service).action_name(request)

在不使用任何容器进行依赖注入的情况下实现这一目标是微不足道的。

我不明白如何使用 php-di

我的objective是通过容器将服务注入控制器,自己将请求对象传递给控制器​​

更新 1

这是我的 ApplicationController

namespace DEC;


class ApplicationController {

    private $service;
    private $request;

    public function __construct(Foo $service, $request) {
        $this->service= $service;
        $this->request = $request;
    }


    public function index() {
        $out = $this->service->foo();
        $out .= $this->request->method();
        return $out;
    }

}

Foo 跟随

namespace DEC;

class Foo {

    public function __construct() {
    }

    public function foo() {
        return "FOO";
    }
}

这是我的请求

namespace DEC;

class Foo {

    public function __construct() {
    }

    public function foo() {
        return "FOO";
    }
}

这是我让 DI 按我的意愿工作的尝试:

$container = ContainerBuilder::buildDevContainer();
$response = $container->call([ApplicationController::class, 'index'], [
            'request' => new Request('GET')
]);
echo $response;

这是我得到的错误:

Entry "DEC\ApplicationController" cannot be resolved: Parameter $request of __construct() has no value defined or guessable
Full definition:
Object (
    class = DEC\ApplicationController
    scope = singleton
    lazy = false
    __construct(
        $service = get(DEC\Foo)
        $request = #UNDEFINED#
    )
)

N.B.: 如果我对请求进行类型提示,错误保持不变 and/or 在构造函数中切换参数的顺序

查看错误,我推断如果我仅使用服务实例化控制器并将请求作为操作方法的参数传递,Matthew Napoli 提出的 ::call() 解决方案就可以工作。

这是否意味着我不能依赖容器进行 "partial" 注入?

更新 2

本次更新中描述的解决方案,请看我自己对问题的回答

不太清楚您尝试了什么,但这应该调用操作方法并将请求传递给它(并解析控制器及其所有依赖项):

$container->call([MyController::class, 'action_name'], [
    'request' => $request,
]);

在此处阅读有关 call() 的更多信息:http://php-di.org/doc/container.html#call

关于请求:

不要对 $request 参数进行类型提示:

public function __construct($request, /*...*/) {}

或者为它输入一个 RequestInterface 提示:

public function __construct(RequestInterface $request, /*...*/) {}

在这两种情况下,DIC 都无法自动创建请求实例。然后你就可以自己注射了。

关于服务:

用具体的类型提示它,比如 "Service"。然后,DIC 将自动创建一个服务实例及其所有依赖项 (data_source)。

或者使用接口类型提示它,例如 "ServiceInterface",并在 DIC 中为它设置一个条目,使用 PHP 定义中的别名。像这样:

return [
    'ServiceInterface' => DI\get('<NAMESPACE-TO>\Service'),
];

参见:

PHP-DI "Limitations" 在 http://php-di.org/doc/autowiring.html#limitations

PHP-DI "PHP definitions - Aliases" 在 http://php-di.org/doc/php-definitions.html#aliases

希望对您有所帮助。

我设法通过在请求控制器之前在容器中设置我的请求来做到这一点:

$container->set('DEC\Request', new Request('GET'));
$controller = $container->get('DEC\ApplicationController');
$response = $controller->index();