JMSSerializer:使用 id 从数据库中检索相关对象
JMSSerializer: Use id to retrieve related object from database
我使用 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 时实现反序列化以检索相关对象?
我考虑过这些可能性:
- 使用特殊的 setter(在
@JMS\Accessor
注释中给出),获取 id 并使用从数据库中检索到的对象填充 subEntity
字段。但这意味着将 EntityManager 注入实体(或类似但不好的东西......)
- 添加一个新的数字字段
subEntityId
并用特殊的 setter 填充它(如上)。然后使用控制器读取它,从数据库中获取 SubEntity
对象并使用 setSubEntity
方法 在 实体被反序列化之后。我也觉得不太好看...
有什么建议吗?我明白了,在 Whosebug 上有一些类似的问题,但在我看来,没有人在描述我的特殊情况。
非常感谢!
将其添加到您的服务配置中,这里我用 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
添加此 class
<?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
}
实际上您不需要创建额外的字段或设置器。可能的方法是创建您自己的 Serializer handler 并在映射中使用它。
我已经回答了 in this topic 所以你可以在那里得到代码示例。
我使用 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 时实现反序列化以检索相关对象?
我考虑过这些可能性:
- 使用特殊的 setter(在
@JMS\Accessor
注释中给出),获取 id 并使用从数据库中检索到的对象填充subEntity
字段。但这意味着将 EntityManager 注入实体(或类似但不好的东西......) - 添加一个新的数字字段
subEntityId
并用特殊的 setter 填充它(如上)。然后使用控制器读取它,从数据库中获取SubEntity
对象并使用setSubEntity
方法 在 实体被反序列化之后。我也觉得不太好看...
有什么建议吗?我明白了,在 Whosebug 上有一些类似的问题,但在我看来,没有人在描述我的特殊情况。
非常感谢!
将其添加到您的服务配置中,这里我用 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
}
实际上您不需要创建额外的字段或设置器。可能的方法是创建您自己的 Serializer handler 并在映射中使用它。
我已经回答了 in this topic 所以你可以在那里得到代码示例。