我必须记录每个实体的更改。我有监听器,它监听教义在preRemove
、postUpdate
和postDelete
的事件。 我的实体AccessModule
有关系:
应用\实体\访问模块.php
/**
* @ORMOneToMany(targetEntity="AppEntityAccessModule", mappedBy="parent")
* @ORMOrderBy({"id" = "ASC"})
*/
private $children;
/**
* @ORMManyToOne(targetEntity="AppEntityAccessModule", inversedBy="children")
* @ORMJoinColumn(name="parent_id", referencedColumnName="id", nullable=true)
*/
private $parent;
/**
* @ORMManyToMany(targetEntity="AppEntityAccessModuleRoute", inversedBy="access_modules")
* @ORMJoinTable(name="access_routes",
* joinColumns={@ORMJoinColumn(name="access_module_id", referencedColumnName="id")},
* inverseJoinColumns={@ORMJoinColumn(name="route_id", referencedColumnName="id")})
*
*/
private $routes;
在侦听器中:App\EventListener\EntityListener.php
use SymfonyComponentSerializerEncoderJsonEncoder;
use SymfonyComponentSerializerEncoderXmlEncoder;
use SymfonyComponentSerializerNormalizerObjectNormalizer;
$encoders = [new XmlEncoder(), new JsonEncoder()];
$normalizer = new ObjectNormalizer();
$normalizer->setCircularReferenceHandler(function ($object) {
return $object->getId();
});
$this->serializer = new Serializer([$normalizer], $encoders);
public function createLog(LifecycleEventArgs $args, $action){
$em = $args->getEntityManager();
$entity = $args->getEntity();
if ($this->tokenStorage->getToken()->getUser()) {
$username = $this->tokenStorage->getToken()->getUser()->getUsername();
} else {
$username = 'anon'; // TODO Remove anon. set null value
}
$log = new Log();
// $log->setData('dddd')
$log->setData($this->serializer->serialize($entity, ''json)
->setAction($action)
->setActionTime(new DateTime())
->setUser($username)
->setEntityClass(get_class($entity));
$em->persist($log);
$em->flush();
}
我在序列化方面遇到问题 当我使用$log->setData($entity)
时,我遇到了循环问题。 我做序列化$log->setData($this->serializer->serialize($entity, ''json)
我充满了关系,与父母的孩子,与孩子的孩子。结果我得到了完整的树:/我想得到
期望
[
'id' => ID,
'name' => NAME,
'parent' => parent_id // ManyToOne, I'd like get its id
'children' => [$child_id, $child_id, $child_id] // array of $id of children array collection
]
(当然这是在将其编码为json之前起草的)
如何在没有完全关系的情况下获得预期数据?
您要查找的内容称为序列化组:此处和此处。
现在让我解释一下它是如何工作的。这很简单。假设您有帖子实体:
class Post
{
/**
* @ORMId
* @ORMGeneratedValue
* @ORMColumn(type="integer")
* @Groups({"default"})
*/
private $id;
/**
* @ORMManyToOne(targetEntity="AppEntityUserUser")
* @Groups({"default"})
*/
private $author;
}
您还有用户实体:
class User
{
/**
* @ORMId
* @ORMGeneratedValue
* @ORMColumn(type="integer")
* @Groups({"default"})
*/
private $id;
/**
* @ORMColumn(type="string", length=40)
* @Groups({"default"})
*/
private $firstName;
/**
* @ORMColumn(type="string", length=40)
*/
private $lastName;
}
帖子可以有作者(用户),但我不想每次都返回所有用户数据。我只对ID和名字感兴趣。
仔细查看@Groups注释。您可以指定所谓的序列化组。这只不过是告诉Symfony您希望在结果集中包含哪些数据的便捷方式。
你必须告诉Symfony序列化程序你想保留哪些关系,方法是在属性/getter上面以注释的形式添加相关组。您还必须指定要保留关系的哪些属性或获取者。
现在如何让Symfony知道这些东西呢?
当您准备/配置序列化服务时,您只需提供如下定义的组:
return $this->serializer->serialize($data, 'json', ['groups' => ['default']]);
围绕本机symfony序列化程序构建某种包装器服务是很好的,这样您就可以简化整个过程并使其更可重用。
还要确保序列化程序配置正确 - 否则它不会考虑这些组。
这也是"处理"循环引用的一种方式(以及其他方式)。
现在,您只需要研究如何设置结果集的格式。
ignored_attributes
提供了一种快速简便的方法来完成您正在寻找的内容。
$serializer->serialize($object, 'json', ['ignored_attributes' => ['ignored_property']]);
以下是它与序列化程序的用法:
use AcmePerson;
use SymfonyComponentSerializerEncoderJsonEncoder;
use SymfonyComponentSerializerNormalizerObjectNormalizer;
use SymfonyComponentSerializerSerializer;
$person = new Person();
$person->setName('foo');
$person->setAge(99);
$normalizer = new ObjectNormalizer();
$encoder = new JsonEncoder();
$serializer = new Serializer([$normalizer], [$encoder]);
$serializer->serialize($person, 'json', ['ignored_attributes' => ['age']]);
文档:https://symfony.com/doc/current/components/serializer.html#ignoring-attributes
在 Symfony 4.1 中测试,以下是实际有效的文档 https://symfony.com/blog/new-in-symfony-2-7-serialization-groups
罗伯特的解释 https://stackoverflow.com/a/48756847/579646 缺少工作$classMetadataFactory。这是我的代码:
$classMetadataFactory = new ClassMetadataFactory(new AnnotationLoader(new AnnotationReader()));
$encoders = [new JsonEncoder()];
$normalizer = new ObjectNormalizer($classMetadataFactory);
$normalizer->setCircularReferenceLimit(2);
// Add Circular reference handler
$normalizer->setCircularReferenceHandler(function ($object) {
return $object->getId();
});
$normalizers = [$normalizer];
$serializer = new Serializer($normalizers, $encoders);
$jsonContent = $serializer->serialize($jobs, 'json', array('groups' => ['default']));
return JsonResponse::fromJsonString($jsonContent);