【问题标题】:Slim Framework : How to break circular dependency injection?Slim 框架:如何打破循环依赖注入?
【发布时间】:2020-08-12 15:55:08
【问题描述】:

我正在使用 Slim 4.5.0 和 PHP-DI 6,但我遇到了循环依赖问题。

我知道如何使用 setter 来解决这类问题,但是在 SlimFramework 的上下文中,我什么都做不了。

我想在发生错误时向 Slack Channel 发送消息。

LoggerInterface::class => function (ContainerInterface $c):Logger
{
  return new Logger($c->get(SlackService::class), $c->get("googleLogger"), (string)$c->get("RCQVersion"), $c->get('settings')['appSettings']['deploymentType'], $c->get('settings')['online']);
},

SecretManagerService::class => function (ContainerInterface $c):SecretManagerService
{
  return new SecretManagerService($c->get('settings'), $c->get(LoggerInterface::class));
},
SlackService::class =>function(ContainerInterface $c):SlackService
{
  $slackToken = $c->get(SecretManagerService::class)->getSecret(SecretManagerService::$SLACK_TOKEN);
  return new SlackService($slackToken, (string)$c->get("RCQVersion"), $c->get('settings')['appSettings']['deploymentType'], $c->get('settings')['online']);
},

我需要为我的自定义 Logger 提供 SlackService。

我尝试在我的 Logger 类中使用 @Inject 关键字来设置 Slack 服务(并将其从构造函数中删除):

  /**
   * @Inject
   * @var SlackService $slackService
   */
  private $slackService;

或者使用setter函数和@Inject(有和没有类名)

 /**
   * @Inject("RedCrossQuest\Service\SlackService")
   * @param SlackService $slackService
   */
  public function setSlackService(SlackService $slackService)
  {
    $this->slackService = $slackService;
  }

但这不起作用,而我觉得这是要走的路。

我已经使用@Inject 为我的类设置属性(字符串值),它运行良好,但在这里,由于某些原因它没有。

我在这里没有找到任何可以帮助理解为什么它不起作用的东西。 https://php-di.org/doc/annotations.html

每次发生错误时,我都会收到一条错误消息,提示 slackService 为 null Uncaught Error: Call to a member function postMessage() on null

让@Inject() 工作我缺少什么?

【问题讨论】:

  • 什么是googlelogger?我不知道我错过了什么,但我看不到循环依赖。
  • Oups,已更正。 Logger 需要 Slack,Slack 需要 SecretManager,SecretManager 需要 Logger
  • SecretManager 是您编写的自定义类吗?它如何以及为什么使用记录器?是不是特别需要SlackService
  • 是的,这是我编写的一个类,用于从 Google Secret Manager 获取秘密。 Slack Service 需要它来获取其 API TOKEN。这里的问题不是如何在功能上打破循环依赖,而是如何在技术上做到这一点。我尝试过@Inject 和通过 Setter 而不是构造函数注入 Slack,但该方法永远不会被调用。我会更新我的帖子
  • 在 github 上查看这些问题 191527 以及 SO 上的 this answer

标签: php dependency-injection slim circular-dependency php-di


【解决方案1】:

正如Nima 与票所指出的那样,循环依赖无法使用setter 解决,除非您使用Lazy Loading。 问题是它需要一个代理库,它有 3 个额外的依赖项,这对于我的简单用例来说有点矫枉过正。 (似乎PHP-DI的文档中缺少一个步骤)

  • zendframework/zend-eventmanager (3.2.1)
  • zendframework/zend-code (3.4.1)
  • ocramius/proxy-manager (2.2.3)
  • ocramius/package-versions (1.5.1)

为了解决这个问题,我手动完成了 PHP-DI 的工作。

  • 我在我的 Logger 上设置了一个 setter 来设置 SlackService,一旦构建了容器,我确实没有在 setter 方法上方的 cmets 中添加 @Inject。
// Set up dependencies
$dependencies = require __DIR__ . '/../../src/dependencies.php';
$dependencies($containerBuilder);

// Build PHP-DI Container instance
$container = $containerBuilder->build();

$loggerInterface = $container->get(LoggerInterface::class);
$loggerInterface->setSlackService ($container->get(SlackService::class);


// Instantiate the app
AppFactory::setContainer($container);
$app = AppFactory::create();

【讨论】:

    猜你喜欢
    • 2016-08-19
    • 1970-01-01
    • 2015-02-12
    • 2020-01-29
    • 1970-01-01
    • 1970-01-01
    • 2014-09-06
    • 1970-01-01
    • 2016-05-24
    相关资源
    最近更新 更多