3

如果它是例如NULL,我想更改实体值。不幸的是,我的代码导致异常 - 尽管在数据库中创建了“翻译”记录并且 id 由 getId 方法正确返回,但似乎没有设置 id。

这是很简单的代码,为什么它不起作用?

public function createAction(Request $request)
{
    $entity  = new Word();
    $form = $this->createForm(new WordType(), $entity);
    $form->bind($request);


    if ($form->isValid()) {

        $em = $this->getDoctrine()->getManager();

        //works fine - database record is created
        if($entity->getIdTranslation() == NULL){
            $translation = new Translation();
            $em->persist($translation);
            $em->flush();
            $entity->setIdTranslation($translation->getId());
        }


        $em->persist($entity);
        //throws exception - Integrity constraint violation: 1048 Column 'id_translation' cannot be null  
        $em->flush();

        return $this->redirect($this->generateUrl('admin_word_show', array('id' => $entity->getId())));
    }

    return array(
        'entity' => $entity,
        'form'   => $form->createView(),
    );
}

编辑:添加了我的模型和映射信息的一部分:

overwiev 模型的一部分

相关词映射:

 /**
 * @var integer $id_language
 *
 * @ORM\Column(name="id_language", type="integer")
 */
private $id_language;
/**
 *@ORM\ManyToOne(targetEntity="Language", inversedBy="words")
 *@ORM\JoinColumn (name="id_language", referencedColumnName="id")
 */
protected $language;

/**
 *
 * @ORM\ManyToOne(targetEntity="Translation", inversedBy="words")
 * @ORM\JoinColumn(name="id_translation", referencedColumnName="id")
 */
protected $translation;

和翻译:

class Translation
 {

public function __construct() {
    $this->words = new ArrayCollection();
}
/**
 * @ORM\PrePersist()
 */
public function prePersist() {
    $this->created_date = new \DateTime();
}
 /**
 *
 * @ORM\OneToMany(targetEntity="Word" ,mappedBy="translation")
 */
protected $words;

另请注意:我使用数据库建模软件(工作台)而不是 Symfony 控制台工具创建了我的数据库模型,现在我正在尝试设置实体以反映数据库模型。我不确定它的方法是否正确——也许我的模型太复杂而无法与 Doctrine 一起正常工作?

4

2 回答 2

9

生命周期回调

有时,您需要在插入、更新或删除实体之前或之后执行操作。这些类型的操作称为“生命周期”回调,因为它们是您需要在实体生命周期的不同阶段(例如,插入、更新、删除实体等)执行的回调方法。

生命周期回调应该是与内部转换实体中的数据有关的简单方法(例如设置创建/更新字段,生成 slug 值)

如果你需要做一些更重的工作——比如执行日志记录或发送电子邮件——你应该注册一个外部类作为事件监听器或订阅者,并让它访问你需要的任何资源。有关详细信息,请参阅如何注册事件侦听器和订阅者。

于 2013-11-24T12:52:32.857 回答
1

一个更好的地方是Word类的构造函数:

class Word
{
    /**
     * @OneToOne(targetEntity="Translation", cascade={"persist"})
     */
    private $translation;

    public function __construct()
    {
        $this->translation = new Translation;
    }
}

顺便说一句,您不需要手动保存翻译 ID,因为它是由 Doctrine 处理的。

$word = new Word;

$em->persist($word);
$em->flush();

$translation = $word->getTranslation();
$translationId = $translation->getId();
于 2012-10-05T06:39:14.303 回答