检测到服务“session”的循环引用,路径:“session -> session.flash_bag -> session”

Posted

技术标签:

【中文标题】检测到服务“session”的循环引用,路径:“session -> session.flash_bag -> session”【英文标题】:Circular reference detected for service "session", path: "session -> session.flash_bag -> session" 【发布时间】:2021-01-07 07:23:38 【问题描述】:

我读过Namespaced Attributes。 我尝试使用此功能:

#src/Controller/CartController.php
public function addProduct(int $cart_id, int $product_id, Request $request, SessionManager $sessionManager)

   $session = new Session(new NativeSessionStorage(), new NamespacedAttributeBag());
   $session->set("cart_$cart_id/product_$product_id", $request->get('selected_product_qty'));
   dump($session);
   die();

效果很好。

所以作为改进,我想override the session services,所以它会在任何地方使用namespacedattributebag,它使用依赖注入而不是将会话服务耦合到addProduct函数中。

为此,我编辑了我的 services.yaml

#config/services.yaml
    session:
        public: true
        class: Symfony\Component\HttpFoundation\Session\Session
        arguments: ['@session.storage', '@session.namespacedattributebag']

    session.namespacedattributebag:
        class: Symfony\Component\HttpFoundation\Session\Attribute\NamespacedAttributeBag

然后我更新CartController

public function addProduct(int $cart_id, int $product_id, Request $request, Session $session)

   $session->set("cart_$cart_id/product_$product_id", $request->get('selected_product_qty'));
   dump($session);
   die();

现在我收到以下错误消息:

这个问题应该已经在这个问题中解决了:#36261 一直是discussed here。

但对我来说问题仍然存在...... (我使用的是 Symfony 5.1)

【问题讨论】:

【参考方案1】:

我无法提供更多解释(很遗憾),但如果出现此问题,您必须明确定义 Session 构造函数的第三个(和可选参数):

#config/services.yaml
session:
    public: true
    class: Symfony\Component\HttpFoundation\Session\Session
    arguments: ['@session.storage', '@session.namespacedattributebag', '@session.flash_bag']

session.namespacedattributebag:
    class: Symfony\Component\HttpFoundation\Session\Attribute\NamespacedAttributeBag

我认为这是因为以下代码:

#src/Symfony/Bundle/FrameworkBundle/DependencyInjection/Compiler/SessionPass.php
foreach ($container->getDefinition('session')->getArguments() as $v) 
    if (!$v instanceof Reference || !isset($bags[$bag = (string) $v]) || !\is_array($factory = $bags[$bag]->getFactory())) 
        continue;
    

    if ([0, 1] !== array_keys($factory) || !$factory[0] instanceof Reference || 'session' !== (string) $factory[0]) 
        continue;
    

    if ('get'.ucfirst(substr($bag, 8, -4)).'Bag' !== $factory[1]) 
        continue;
    
    $bags[$bag]->setFactory(null);

如果你调试它,你会看到:

dump($bags);
foreach($container->getDefinition('session')->getArguments() as $v)...
dump($bags);

你的配置是这样的:

#config/services.yaml
...
arguments: ['@session.storage', '@session.namespacedattributebag', '@session.flash_bag']
...

你会注意到调试显示:

"session.flash_bag" => Symfony\Component\DependencyInjection\Definition #2253 ▼
    -class: "Symfony\Component\HttpFoundation\Session\Flash\FlashBag"
    -file: null
    -factory: array:2 [▶]

对于flash_bag的输入服务Definition 然后在输出中(第二个dump()):

"session.flash_bag" => Symfony\Component\DependencyInjection\Definition #2253 ▼
    -class: "Symfony\Component\HttpFoundation\Session\Flash\FlashBag"
    -file: null
    -factory: null

factory 参数为空。

所以我猜在这种情况下flash_bag 对象没有被创建,所以没有Circular reference...

我无法解释更多。

另外,Symfony 工作人员的某个人说: @warslett commented on 28 Mar

@ekosogin 你实际上并不需要 @session.flash_bag 参数 那里。你应该可以这样做:

session:
    class: Symfony\Component\HttpFoundation\Session\Session
    arguments:
        - '@session.storage'
        - '@Symfony\Component\HttpFoundation\Session\Attribute\NamespacedAttributeBag'
Symfony\Component\HttpFoundation\Session\Attribute\NamespacedAttributeBag:
~```

但实际上,我认为这是错误的,因为如果您不按照他所说的那样放置@session.flash_bag 参数,它将无法正常工作。

希望对你有所帮助。

【讨论】:

以上是关于检测到服务“session”的循环引用,路径:“session -> session.flash_bag -> session”的主要内容,如果未能解决你的问题,请参考以下文章

如何解决检测到的循环引用

检测到实体框架自引用循环[重复]

在 ASP.NET Core 中检测到自引用循环 [重复]

检测到自引用循环 - 从 WebApi 取回数据到浏览器

Kendo UI - InvalidOperationException:序列化类型对象时检测到循环引用

为 WebApi 2 中的属性检测到自引用循环