JMSSerializer:使用id从数据库中检索相关对象

时间:2017-02-23 14:37:45

标签: json doctrine-orm symfony jmsserializerbundle

使用Symfony 3我编写了一个应该通过JSON / JMSSerializer公开的实体(以RESTful方式)。它看起来像这样:

/**
 * MainEntity
 *
 * @ORM\Table(name="MainEntity")
 * @ORM\Entity
 * 
 */
 class MainEntity{

   /**
    * @var integer
    *
    * @ORM\Column(name="id", type="integer", nullable=false)
    * @ORM\Id
    * @ORM\GeneratedValue(strategy="IDENTITY")
    */
   private $id;

   //... some more "simple" fields ...

   /**
   * @ORM\ManyToOne(targetEntity="SubEntity")
   * @ORM\JoinColumns({
   *   @ORM\JoinColumn(name="subentity", referencedColumnName="id")
   * })
   * @JMS\Accessor(getter="getSubEntityId",setter="setSubEntity")
   * @JMS\Type("integer")
   * @JMS\SerializedName("subEntityId")
   */
   private $subEntity;
   //...

   public function getSubEntityId() {
       return $this->subEntity->getId();
   }

}
整个事情的JSON序列化就像一个魅力!特别是,只暴露了子实体ID,而不是整个子实体,它可能非常大。

所以而不是:

{"id": 1, ..., "subEntity": {"id": 123, "name": "Great subEntity", ...} }

我得到了

{"id": 1, ..., "subEntityId": 123 }

这完全是我所需要的。

但是当谈到 de 序列化时,我遇到了麻烦...当然我也希望在传入的请求中使用缩短的JSON格式,但这会失败,因为setSubEntity期望SubEntity实例而不是数字。

在给出ID时,有什么办法可以实现反序列化来检索相关对象吗?

我想到了这些可能性:

  1. 使用特殊的setter(在@JMS\Accessor注释中给出),获取id并使用从数据库中检索的对象填充subEntity字段。但这意味着将EntityManager注入实体(或类似的东西......)
  2. 添加一个新的数字字段subEntityId,并通过特殊的setter填充(如上所示)。然后使用控制器读取它,从数据库中获取SubEntity对象,并在反序列化实体后使用setSubEntity方法。对我来说看起来也不那么好......
  3. 有什么建议吗?我知道,Stackoverflow上有一些类似的问题,但在我看来,没有人描述我的特殊情况。

    非常感谢提前!

2 个答案:

答案 0 :(得分:1)

实际上,您不需要创建额外的字段或设置者。可能的方法是创建自己的Serializer handler并在映射中使用它。

我已经回复了in this topic,因此您可以在那里获取代码示例。

答案 1 :(得分:0)

将此添加到您的服务配置中,此处我将其写入YAML

jms_serializer.object_constructor:
    alias: jms_serializer.initialized_object_constructor
    public: false

jms_serializer.initialized_object_constructor:
    class: MyApp\Bundle\CoreBundle\Serializer\InitializedObjectConstructor
    arguments: ["@jms_serializer.unserialize_object_constructor"]

jms_serializer.initialized_object_constructor

添加此课程
<?php
namespace MyApp\Bundle\CoreBundle\Serializer;
use JMS\Serializer\VisitorInterface;
use JMS\Serializer\Metadata\ClassMetadata;
use JMS\Serializer\DeserializationContext;
use JMS\Serializer\Construction\ObjectConstructorInterface;
class InitializedObjectConstructor implements ObjectConstructorInterface
{
    private $fallbackConstructor;
    public function __construct(ObjectConstructorInterface $fallbackConstructor)
    {
        $this->fallbackConstructor = $fallbackConstructor;
    }
    public function construct(VisitorInterface $visitor, ClassMetadata $metadata, $data, array $type, DeserializationContext $context)
    {
        if ($context->attributes->containsKey('target') && $context->getDepth() === 1) {
            return $context->attributes->get('target')->get();
        }
        return $this->fallbackConstructor->construct($visitor, $metadata, $data, $type, $context);
    }
}

在您的控制器中:

protected function flushRequestData(Request $request, $entity = null)
{
    $data = $request->getContent();
    $dm = $this->get('doctrine_mongodb.odm.default_document_manager');
    $context = new DeserializationContext();
    if ($entity) {
        $context->attributes->set('target', $entity);
    }

    $deserializedObj = $this->get('serializer')->deserialize(
        $data,
        $this->getRepository()->getClassName(),
        'json',
        $context
    );

    // After deserialized into your entity you need to manually set each of the related entity one by one manually
}