如何将实体另存为《 Doctrine 2》中的另一行


77

假设我有实体$e。是否有任何通用方法将其存储为另一行,该行将具有相同的实体数据但具有另一个主键?

为什么需要这样做:我正在实现某种临时数据库模式,而不是更新行,而只需要创建另一行。


就在我头顶上(即未经测试),您尝试过$f = clone $e吗?您可能需要实现__clone()方法
菲尔

4
@Phil:克隆的实体具有相同的PK,因此仅更新同一行。甚至更令人惊讶的是- spl_object_hash(Doctrine使用它来标识特定的实例)即使原始对象和克隆对象包含不同的数据,它们也是相同的
zerkms 2012年

@Phil:__clone()也无济于事-Doctrine使用$oid = spl_object_hash($entity);和一些内部映射来获取对象的状态。对于原始MANAGED
副本

1
这不是真的。克隆$ e返回另一个实例,并因此返回另一个spl_object_hash()值。
Florian Klein

1
试过了,还是可以肯定的。克隆是一个不同的实例,在您要求UnitOfWork / identityMap对其进行注册之前,该实体将被视为已插入。
弗洛里安·克莱因

Answers:


161

尝试克隆并将以下方法添加到您的实体中

public function __clone() {
    $this->id = null;
}

您可能需要先分离实体,然后再保留它。我现在没有开发机可以方便地对此进行测试。

$f = clone $e;
$em->detach($f);
$em->persist($f);
$em->flush();

更新资料

刚刚尝试使用简单的SQLite演示。您不需要做任何事情。以下对我有用,而无需添加__clone()方法或执行其他任何异常操作

$new = clone $old;
$em->persist($new);
$em->flush();

刷新后,该$new实体具有新的ID,并保存为数据库中的新行。

我仍然会通过该__clone()方法将ID属性设为空,因为从纯模型视图来看,它是有意义的。

更新2

深入研究Doctrine代码,这是因为生成的代理类__clone()使用这一重要行实现

unset($this->_entityPersister, $this->_identifier);

6
当我尝试此方法时,对原始实体的任何更改也将保留到数据库中,并插入新记录。这是因为实体管理器仍在对其进行管理。即使当我分离原始实体时,它仍然保留更改。我不知道为什么,或者不知道如何成功让实体管理器停止管理原始实体(即放弃所做的更改)。都不起作用$em->refresh($old)$em->detach($old)似乎不起作用...
Chadwick Meyer

2
如果要实施,请__clone()确保安全地执行该操作,如文档中所示,否则您可能会轻易中断事情。
Yep_It's_Me 2015年

2
请注意,实现__clone()方法仅在平面对象没有任何关系的情况下才有效。特别是,仅复制不是对其他对象的引用的数据,所有引用将与原始对象中的相同。
barbieswimcrew

2
@barbieswimcrew正确。对于这种情况,我还有另一个答案
菲尔

2
@ Yep_It's_Me这是一个完整的(atm)链接,指向有关如何实现克隆或唤醒的
Dimitry K


-3

这是我使用的简单策略,不涉及过多的复杂性:

$new->fromArray( $old->toArray() );
$new->id = NULL;

2
在哪里fromArraytoArray方法从何而来?
zerkms

您不使用Doctrine_Collection对象吗?
redolent

1
我只使用普通的php对象,该对象仅保存数据并提供访问器/更改器方法来访问/修改它。它没有扩展任何教义课程
zerkms,2013年
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.