我一直在想是否可以使用JMS序列化器将JSON反序列化为现有对象。
通常,这对于用JSON格式的新数据更新现有对象很有用。 Symfony的标准反序列化器似乎可以提供这种功能,但是我似乎无法在JMS中找到任何相关信息。如果我想要serializedName Annotation选项,则必须使用JMS。
“解决方法”是反序列化,然后使用Doctrine的EntityManager进行合并,但效果如此之好,如果JSON不包含每个字段,您就无法轻易辨别要更新的字段。
答案 0 :(得分:1)
所以可以做到这一点,我还没有完全弄清楚如何做,但是我又放弃了JMS,因此仅供参考,因为我想这比无缘无故地打开问题要好:
https://github.com/schmittjoh/serializer/issues/79,您可能还会在GitHub上找到更多挖掘机会。
答案 1 :(得分:0)
我一直在努力寻找解决方案,但最终找到了解决方案,然后我们开始:
services.yaml
jms_serializer.object_constructor:
alias: jms_serializer.initialized_object_constructor
jms_serializer.initialized_object_constructor:
class: App\Service\InitializedObjectConstructor
arguments: ["@jms_serializer.unserialize_object_constructor"]
App\Service\InitializedObjectConstructor.php
<?php
declare(strict_types=1);
namespace App\Service;
use JMS\Serializer\Construction\ObjectConstructorInterface;
use JMS\Serializer\DeserializationContext;
use JMS\Serializer\Metadata\ClassMetadata;
use JMS\Serializer\Visitor\DeserializationVisitorInterface;
class InitializedObjectConstructor implements ObjectConstructorInterface
{
private $fallbackConstructor;
/**
* @param ObjectConstructorInterface $fallbackConstructor Fallback object constructor
*/
public function __construct(ObjectConstructorInterface $fallbackConstructor)
{
$this->fallbackConstructor = $fallbackConstructor;
}
/**
* {@inheritdoc}
*/
public function construct(
DeserializationVisitorInterface $visitor,
ClassMetadata $metadata,
$data,
array $type,
DeserializationContext $context
): ?object {
if ($context->hasAttribute('target') && 1 === $context->getDepth()) {
return $context->getAttribute('target');
}
return $this->fallbackConstructor->construct($visitor, $metadata, $data, $type, $context);
}
}
$object = $this->entityManager->find('YourEntityName', $id);
$context = new DeserializationContext();
$context->setAttribute('target', $object);
$data = $this->serializer->deserialize($request->getContent(), 'YourEntityClassName', 'json', $context);