问题 如何正确地将依赖注入Laravel工匠命令?


基本上我想从laravel命令调用存储库Repository.php上的方法。

Example\Storage\Repository.php
Example\Storage\RepositoryInerface.php
Example\Storage\RepositoryServiceProvider.php

我希望在命令构造函数中使用Interface,然后将其设置为受保护的变量。

在服务提供者中,我将Interface绑定到Repository类。

现在,在start / artisan.php中我写道:

Artisan::add(new ExampleCommand(new Repository());

我可以在这里使用界面吗?什么是正确的方法?我很困惑。

提前致谢。

编辑:澄清一下,它只能按照现在的方式工作,但我不想在注册artisan命令时硬编码具体类。


4485
2018-03-14 11:03


起源



答案:


您可以使用IoC容器的自动依赖注入功能:

Artisan::add(App::make('\Example\Commands\ExampleCommand'));
// or
Artisan::resolve('\Example\Commands\ExampleCommand');

如果ExampleCommand的构造函数接受一个具体类作为其参数,那么它将自动注入。如果它依赖于接口,则需要告诉IoC容器在请求给定接口时使用特定的具体类。

具体(为简洁而忽略名称空间):

class ExampleCommand ... {
    public function __construct(Repository $repo) {
    }
}

Artisan::resolve('ExampleCommand');

接口(为简洁而忽略名称空间):

class ExampleCommand ... {
    public function __construct(RepositoryInterface $repo) {
    }
}

App::instance('RepositoryInterface', new Repository);
Artisan::resolve('ExampleCommand');

16
2018-03-14 11:12



Command构造函数接受Repository接口,命令本身不在IoC中只是Repository,我需要注入到位于默认目录中的命令。 - Ren
我不明白的是我需要改变什么才能做到 Artisan::add(new ExampleCommand(new Repository()); 改成 Artisan::add(new ExampleCommand(new RepositoryInterface()); - Ren
你不需要IoC容器中的命令 - IoC容器很聪明,如果它得到一个尚未在容器中的'classname',它会尝试实例化它 和 它的构造函数的依赖项。试试我的代码并查看,确保您使用FQCN。 - alexrussell
这真的很神奇。试着 Artisan::add(App::make('\Example\Commands\ExampleCommand')); 它按预期工作。我在哪里可以阅读更多相关信息?你能详细说明吗? :) - Ren
它解释得非常好 在文档中。我认为这是一种方便的自动分辨率副作用 App::make() 任何课程。关键是当IoC容器需要进行自动解析时,它所知道的只是一个构造函数的参数,从中可以得出提示中的类,并从那里找出如何创建要传入的实例。所以,相反而不是为这个分辨率创建一个受保护的方法,泰勒成为了这个分辨率 make() 方法,所以任何代码都可以使用它,而不仅仅是IoC容器。 - alexrussell


答案:


您可以使用IoC容器的自动依赖注入功能:

Artisan::add(App::make('\Example\Commands\ExampleCommand'));
// or
Artisan::resolve('\Example\Commands\ExampleCommand');

如果ExampleCommand的构造函数接受一个具体类作为其参数,那么它将自动注入。如果它依赖于接口,则需要告诉IoC容器在请求给定接口时使用特定的具体类。

具体(为简洁而忽略名称空间):

class ExampleCommand ... {
    public function __construct(Repository $repo) {
    }
}

Artisan::resolve('ExampleCommand');

接口(为简洁而忽略名称空间):

class ExampleCommand ... {
    public function __construct(RepositoryInterface $repo) {
    }
}

App::instance('RepositoryInterface', new Repository);
Artisan::resolve('ExampleCommand');

16
2018-03-14 11:12



Command构造函数接受Repository接口,命令本身不在IoC中只是Repository,我需要注入到位于默认目录中的命令。 - Ren
我不明白的是我需要改变什么才能做到 Artisan::add(new ExampleCommand(new Repository()); 改成 Artisan::add(new ExampleCommand(new RepositoryInterface()); - Ren
你不需要IoC容器中的命令 - IoC容器很聪明,如果它得到一个尚未在容器中的'classname',它会尝试实例化它 和 它的构造函数的依赖项。试试我的代码并查看,确保您使用FQCN。 - alexrussell
这真的很神奇。试着 Artisan::add(App::make('\Example\Commands\ExampleCommand')); 它按预期工作。我在哪里可以阅读更多相关信息?你能详细说明吗? :) - Ren
它解释得非常好 在文档中。我认为这是一种方便的自动分辨率副作用 App::make() 任何课程。关键是当IoC容器需要进行自动解析时,它所知道的只是一个构造函数的参数,从中可以得出提示中的类,并从那里找出如何创建要传入的实例。所以,相反而不是为这个分辨率创建一个受保护的方法,泰勒成为了这个分辨率 make() 方法,所以任何代码都可以使用它,而不仅仅是IoC容器。 - alexrussell


你可以使用 interface 在构造函数中键入提示所依赖的对象,但您必须将具体类绑定到该接口中 IoC 容器使用类似下面的东西,所以它会工作。

App::bind('Example\Storage\RepositoryInerface', 'Example\Storage\Repository');

阅读更多内容 文件


0
2018-03-14 11:08



是的我已经在ServiceProvider中这样做了,它在Command构造函数中正常工作,但是如何通过接口 Artisan::add(new ExampleCommand(new Repository()); 这里? - Ren
你不需要通过它, IoC 容器将自动注入依赖项。 - The Alpha
所以我必须在这里具体课?那么绑定对我有什么帮助?如果我使用接口实现另一个类,我将需要更改绑定+在此更改它。我错过了什么?请参阅我对其他答案的评论。 - Ren