在 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 的环境,例如 dev 或 test。不好吧?
我从 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 继承来设置 public、autowire 和 如果您在服务声明中使用 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。
我正在将一个 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 中的包(我知道什么你会说这个,但我打算尽快升级所有这些)。
我的问题有点不同,这就是为什么
我从 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 继承来设置 public、autowire 和 如果您在服务声明中使用 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。