in7*_*n7l 11 php symfony doctrine-orm
过去几天我一直在研究这个问题而没有太大成功,但我偶然发现了这个与我想要实现的目标非常相关的答案.
我已注册并添加了loadClassMetadata侦听器,该侦听器映射字段foo:
// src/DynamicMappingTest/AdminBundle/EventListener/MappingListener.php
namespace DynamicMappingTest\AdminBundle\EventListener;
use Doctrine\ORM\Event\LoadClassMetadataEventArgs;
class MappingListener
{
public function loadClassMetadata(LoadClassMetadataEventArgs $eventArgs)
{
$classMetadata = $eventArgs->getClassMetadata();
if ($classMetadata->getName() != 'DynamicMappingTest\\AdminBundle\\Entity\\CustomNode')
{
// Not the CustomNode test class. Do not alter the class metadata.
return;
}
$table = $classMetadata->table;
$oldName = $table['name']; // ... or $classMetaData->getTableName()
// your logic here ...
$table['name'] = 'custom_node';
$classMetadata->setPrimaryTable($table);
$reflClass = $classMetadata->getReflectionClass();
dump($reflClass);
// ... or add a field-mapping like this
$fieldMapping = array(
'fieldName' => 'foo',
'type' => 'string',
'length' => 255
);
$classMetadata->mapField($fieldMapping);
}
}
Run Code Online (Sandbox Code Playgroud)
现在,只要我在DynamicMappingTest\AdminBundle\Entity\CustomNode类中声明了foo属性,这一切都有效:
// src/DynamicMappingTest/AdminBundle/Entity/CustomNode.php
namespace DynamicMappingTest\AdminBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
/**
* CustomNode
*
* @ORM\Table()
* @ORM\Entity(repositoryClass="DynamicMappingTest\AdminBundle\Entity\CustomNodeRepository")
*/
class CustomNode
{
...
private $foo;
}
Run Code Online (Sandbox Code Playgroud)
但是,我无法知道用户将为其自定义实体定义哪些属性.如果我删除FOO从属性CustomNode班,ReflectionClass,我从了ClassMetadata得到自然不会包括富财产,所以我得到每当下面的异常MapField可()在MappingListener执行:
ReflectionException: Property DynamicMappingTest\AdminBundle\Entity\CustomNode::$foo does not exist
in vendor/doctrine/common/lib/Doctrine/Common/Persistence/Mapping/RuntimeReflectionService.php at line 80
Run Code Online (Sandbox Code Playgroud)
77. */
78. public function getAccessibleProperty($class, $property)
79. {
80. $reflectionProperty = new ReflectionProperty($class, $property);
81.
82. if ($reflectionProperty->isPublic()) {
83. $reflectionProperty = new RuntimePublicReflectionProperty($class, $property);
Run Code Online (Sandbox Code Playgroud)
Doctrine 为您的实体生成代理类。这意味着学说会生成带有类的 PHP 代码,该类扩展您的实体类并覆盖方法 - 放置一些自定义逻辑,然后调用父方法。
因此,我认为实现这一目标的唯一方法是为代码中的实体生成 PHP 代码。也就是说,每次在您的网站中创建实体时,您都应该使用该实体生成 PHP 文件,然后运行迁移。
我认为在这种情况下您根本不应该使用 Doctrine ORM,至少不应该以您尝试这样做的方式。
一般来说,ORM 用于更容易/更易于管理的编程。也就是说,您可以设置关系、使用延迟加载、工作单元(更改实体属性然后刷新)等。如果您的实体是动态生成的,您将使用哪些功能?开发人员不会为这些实体编写代码,因为正如您所说,无法知道它将具有哪些字段。
您还没有提供具体的用例 - 为什么您首先要这样做。但我想这确实可以通过某种更简单的方式来完成。
如果用户可以存储任何结构,你还应该使用 MySQL 吗?在这种情况下,ElasticSearch 或类似的解决方案可能会更好。
正如我所提到的 - 是的。除非你想覆盖或替换一些 Doctrine 代码,但我想它可能有很多(代理类等)
| 归档时间: |
|
| 查看次数: |
2022 次 |
| 最近记录: |