在 Symfony 3.3 中使用 DI 进行抽象 类 自动装配,这可能吗?

Autowiring in abstract classes with DI in Symfony 3.3, is it possible?

我正在将一个 Symfony 3.2 项目移动到 Symfony 3.3,我想使用 DI new features. I have read the docs 但到目前为止我可以让它工作。请参阅以下 class 定义:

use Http\Adapter\Guzzle6\Client;
use Http\Message\MessageFactory;

abstract class AParent
{
    protected $message;
    protected $client;
    protected $api_count_url;

    public function __construct(MessageFactory $message, Client $client, string $api_count_url)
    {
        $this->message       = $message;
        $this->client        = $client;
        $this->api_count_url = $api_count_url;
    }

    public function getCount(string $source, string $object, MessageFactory $messageFactory, Client $client): ?array
    {
        // .....
    }

    abstract public function execute(string $source, string $object, int $qty, int $company_id): array;
    abstract protected function processDataFromApi(array $entities, int $company_id): array;
    abstract protected function executeResponse(array $rows = [], int $company_id): array;
}

class AChildren extends AParent
{
    protected $qty;

    public function execute(string $source, string $object, int $qty, int $company_id): array
    {
        $url      = $this->api_count_url . "src={$source}&obj={$object}";
        $request  = $this->message->createRequest('GET', $url);
        $response = $this->client->sendRequest($request);
    }

    protected function processDataFromApi(array $entities, int $company_id): array
    {
        // ....
    }

    protected function executeResponse(array $rows = [], int $company_id): array
    {
        // ....
    }
}

这是我的 app/config/services.yml 文件的样子:

parameters:
    serv_api_base_url: 'https://url.com/api/'

services:
    _defaults:
        autowire: true
        autoconfigure: true
        public: false

    CommonBundle\:
        resource: '../../src/CommonBundle/*'
        exclude: '../../src/CommonBundle/{Entity,Repository}'

    CommonBundle\Controller\:
        resource: '../../src/CommonBundle/Controller'
        public: true
        tags: ['controller.service_arguments']

    # Services that need manually wiring: API related
    CommonBundle\API\AParent:
        arguments:
            $api_count_url: '%serv_api_base_url%'

但我收到以下错误:

AutowiringFailedException Cannot autowire service "CommonBundle\API\AChildren": argument "$api_count_url" of method "__construct()" must have a type-hint or be given a value explicitly.

当然,我在这里遗漏了一些东西,或者只是这是不可能的,这让我想到了下一个问题:这是一个糟糕的 OOP 设计还是 Symfony 3.3 DI 功能中缺少的功能?

当然我不想让 AParent class 成为一个接口,因为我不想重新定义 classes 上实现这种接口的方法。

我也不想重复自己和copy/paste相同的功能children。

想法?线索?建议?这可能吗?

更新

阅读“How to Manage Common Dependencies with Parent Services”后,我在我的场景中尝试了以下操作:

CommonBundle\API\AParent:
    abstract: true
    arguments:
        $api_count_url: '%serv_api_base_url%'

CommonBundle\API\AChildren:
    parent: CommonBundle\API\AParent
    arguments:
        $base_url: '%serv_api_base_url%'
        $base_response_url: '%serv_api_base_response_url%' 

但是错误变成:

Attribute "autowire" on service "CommonBundle\API\AChildren" cannot be inherited from "_defaults" when a "parent" is set. Move your child definitions to a separate file or define this attribute explicitly in /var/www/html/oneview_symfony/app/config/services.yml (which is being imported from "/var/www/html/oneview_symfony/app/config/config.yml").

但是我可以让它与以下设置一起工作:

CommonBundle\API\AParent:
    arguments:
        $api_count_url: '%serv_api_base_url%'

CommonBundle\API\AChildren:
    arguments:
        $api_count_url: '%serv_api_base_url%'
        $base_url: '%serv_api_base_url%'
        $base_response_url: '%serv_api_base_response_url%'

这是正确的方法吗?有道理吗?

更新 #2

按照@Cerad 的说明,我制作了一些模组(参见上面的代码和下面的定义),现在 objects 即将到来 NULL?知道这是为什么吗?

// services.yml
services:
    CommonBundle\EventListener\EntitySuscriber:
        tags:
            - { name: doctrine.event_subscriber, connection: default}

    CommonBundle\API\AParent:
        abstract: true
        arguments:
            - '@httplug.message_factory'
            - '@httplug.client.myclient'
            - '%ser_api_base_url%'

// services_api.yml
services:
    CommonBundle\API\AChildren:
        parent: CommonBundle\API\AParent
        arguments:
            $base_url: '%serv_api_base_url%'
            $base_response_url: '%serv_api_base_response_url%'

// config.yml
imports:
    - { resource: parameters.yml }
    - { resource: security.yml }
    - { resource: services.yml }
    - { resource: services_api.yml }

为什么 objects 在 child class 中是 NULL

我认为答案就在错误消息中,与使用摘要无关class。 您是否以这样的方式设置了自动装配,以便 AChildren 无需您的明确指示即可接通?如果是这样,您可能需要为每个子 class 指定构造函数参数。 请参阅 https://symfony.com/doc/current/service_container/parent_services.html 以获得可能的帮助

有意思。您似乎希望自动装配了解 AChild 扩展 AParent,然后使用 AParent 服务定义。我不知道这种行为是被故意忽略还是设计不支持。 autowire 仍处于起步阶段,正在大量开发中。

我建议前往 di github 存储库,检查问题,然后在适用的情况下打开一个。开发人员会告诉您这是否是设计使然。

同时,如果将子定义移动到不同的服务文件,则可以使用父服务功能。它会起作用,因为 _defaults 内容仅适用于当前服务文件。

# services.yml
AppBundle\Service\AParent:
    abstract: true
    arguments:
        $api_count_url: '%serv_api_base_url%'

# services2.yml NOTE: different file, add to config.yml
AppBundle\Service\AChild:
    parent: AppBundle\Service\AParent
    arguments:
        $base_url: 'base url'

最后一个稍微偏离主题的注释:没有必要 public: false 除非你玩弄自动配置的东西。默认情况下,所有服务都定义为私有的,除非您明确声明它们是 public.

更新 - 一条评论提到了有关对象为空的内容。不太确定那是什么意思,但我去并在我的测试中添加了一个记录器 类。所以:

use Psr\Log\LoggerInterface;

abstract class AParent
{
    protected $api_count_url;

    public function __construct(
        LoggerInterface $logger, 
        string $api_count_url)
    {
        $this->api_count_url = $api_count_url;
    }
}    
class AChild extends AParent
{
    public function __construct(LoggerInterface $logger, 
        string $api_count_url, string $base_url)
    {
        parent::__construct($logger,$api_count_url);
    }

并且由于只有一个 psr7 记录器实现,记录器是自动装配和注入的,无需更改服务定义。

更新2 我更新到 S3.3.8 并开始获得:

[Symfony\Component\DependencyInjection\Exception\RuntimeException]                                                                         
Invalid constructor argument 2 for service "AppBundle\Service\AParent": argument 1 must be defined before. Check your service definition.  

Autowire 仍在大力开发中。在这一点上不打算花精力去弄清楚为什么。与参数的顺序有关。一旦 LTS 版本发布,我将重新发布。

自动装配似乎不适用于 Symfony 3.3.8 中的抽象服务。我为此创建 an issue on Github

与此同时,我个人删除了 abstract 选项并且它工作正常。

好吧,我遇到了同样的问题。出于某些原因,我不得不创建一个 Symfony 3.3.17 项目来启动,因为在我的公司我不得不使用我们的 2 个仍然停留在 SF 3.3 中的包(我知道什么你会说这个,但我打算尽快升级所有这些)。

我的问题有点不同,这就是为什么 was a little bit complicated to use in my case. But I can bring (maybe) also a solution for the issue mentioned in the question. In my case, I am using Symfony Flex 管理我的应用程序,即使它是 SF 3.3。因此,了解 flex 的人都知道 config.yml 不再存在,取而代之的是 config 文件夹位于应用。在其中,您只有一个 services.yaml 文件。所以,如果你想添加一个 services2.yaml 文件,你会发现它没有被检测到,除非你将它重命名为 services_2.yaml。但在这种情况下,这意味着您有另一个名为 2 的环境,例如 devtest。不好吧?

我从 xabbuh's answer in this issue 中找到了解决方案。

因此,为了能够在 Symfony 3.3 中 抽象地使用自动装配 类 和 DI 你仍然可以将你的服务定义保存在一个文件中:

CommonBundle\API\AParent:
    abstract: true
    autoconfigure: false
    arguments:
        $api_count_url: '%serv_api_base_url%'

CommonBundle\API\AChildren:
    parent: CommonBundle\API\AParent
    autoconfigure: false
    autowire: true
    public: false
    arguments:
        $base_url: '%serv_api_base_url%'
        $base_response_url: '%serv_api_base_response_url%'

这里的重点是

you need to be a bit more explicit

正如xabbuh所说。您不能从 _default 继承来设置 publicautowire如果您在服务声明中使用 parent 键,则自动配置。将 child 服务中的 autoconfigure 值设置为 false 很重要,因为如果未设置,您将拥有

The service "CommonBundle\API\AChildren" cannot have a "parent" and also have "autoconfigure". Make sense in fact...

最后一件事,如果问题与 Symfony 控制台命令有关(就像我的情况),请不要忘记使用

tags:
    - { name: console.command }

用于您的 child 服务,因为 autoconfigure 设置为 false