当前位置: 首页 > 知识库问答 >
问题:

Symfony原则2 PostRemove删除文件奇怪的行为

柳杰
2023-03-14

参考Symfony 2关于文件上传的Cookbook,我尝试使用Doctrine的@PostRemove事件监听器在文件从数据库中删除后删除文件。

Document.php

/** @Entity */
class Document {
    /** OneToOne(targetEntity="File", cascade={"all"}) */
    private $file;

    public function setFile(File $file) {
        $this->file = $file;
    }

    public function getFile() {
        return $this->file;
    }
}

File.php

/*
** @Entity
** @HasLifecycleCallbacks
*/
class File extends {
    /** @Column(type="string") */
    private $name;

    public function __construct(UploadedFile $file) {
        $this->path = $file->getPathname();
        $this->name = $file->getClientOriginalName();
    }

    public function getAbsolute() {
        return '/var/www/cdn.myweb.com/file/'.$this->name;
    }

    /** @PostRemove */
    public function removeFile() {
        unlink($this->getAbsolute());
    }
}

数据库:

**Document**
--------
|  id  |
-------
|   1  |
--------

**DocumentFiles**
--------------------------
| document_id | file_id  |
--------------------------
|      1      |     2    |
--------------------------

**File**
--------------------
|  id  |   name    |
--------------------
|   1  | file1.ext |
|   2  | file2.ext |
--------------------

当我删除ID为1的文档时,不知何故学说也解除关联ID为1的文件。

据我所知,这些步骤导致了这种奇怪的行为:

1-条令的UnitOfWork将调用commit()方法,该方法反过来调用executeElections()

2-在执行删除()中,参与者根据文档的id删除了文档的文件,然后执行

if ( ! $class->isIdentifierNatural()) {
    $class->reflFields[$class->identifier[0]]->setValue($entity, null);
}

它将文件的id值设置为null,然后开始调用其事件

if ($invoke !== ListenersInvoker::INVOKE_NONE) {
    $this->listenersInvoker->invoke($class, Events::postRemove, $entity, new LifecycleEventArgs($entity, $this->em), $invoke);
}

3-由于文档模型中的$file属性是一对一的关系,所以条令自动创建一个文件代理类来代替真实的文件类作为文档的属性。

4-调用removeFile()函数时,它调用FileProxy的getAbsolutePath()

FileProxy.php

public function getAbsolute() {
    $this->__initializer__ && $this->__initializer__invoke($this, 'getAbsolute', array());

    return parent::getAbsolute();
}

它调用带有闭包的初始化器:

    function (BaseProxy $proxy) use ($entityPersister, $classMetadata) {
        $initializer = $proxy->__getInitializer();
        $cloner      = $proxy->__getCloner();

        $proxy->__setInitializer(null);
        $proxy->__setCloner(null);

        if ($proxy->__isInitialized()) {
            return;
        }

        $properties = $proxy->__getLazyProperties();

        foreach ($properties as $propertyName => $property) {
            if (!isset($proxy->$propertyName)) {
                $proxy->$propertyName = $properties[$propertyName];
            }
        }

        $proxy->__setInitialized(true);

        if (null === $entityPersister->load($classMetadata->getIdentifierValues($proxy), $proxy)) {
            $proxy->__setInitializer($initializer);
            $proxy->__setCloner($cloner);
            $proxy->__setInitialized(false);

            throw new EntityNotFoundException();
        }
    };

5-调用$entityPersister-

public function load(array $criteria, $entity = null, $assoc = null, array $hints = array(), $lockMode = 0, $limit = null, array $orderBy = null)
{
    $sql = $this->getSelectSQL($criteria, $assoc, $lockMode, $limit, null, $orderBy);
    list($params, $types) = $this->expandParameters($criteria);
    $stmt = $this->conn->executeQuery($sql, $params, $types);

    if ($entity !== null) {
        $hints[Query::HINT_REFRESH]         = true;
        $hints[Query::HINT_REFRESH_ENTITY]  = $entity;
    }

    $hydrator = $this->em->newHydrator($this->selectJoinSql ? Query::HYDRATE_OBJECT : Query::HYDRATE_SIMPLEOBJECT);
    $entities = $hydrator->hydrateAll($stmt, $this->rsm, $hints);

    return $entities ? $entities[0] : null;
}

6-EntityPermet的load()将水合FileProxy与它的查询结果中的最后一项,这是带有Id 1的文件,因为根据当前事务带有Id 2的文件已经被删除。

我在文档的$file映射中使用了fetch=EAGER,解决了这个问题,但我对这个问题很好奇。

我做错什么了吗?这是一种预期的行为,还是一个bug?


共有1个答案

韦锦程
2023-03-14

Fatech=EAGER工作正常,但是在我们的例子中,我们有很多依赖于图像实体的实体,所以我们在preRemove中添加了初始化,而不是在所有实体中添加Fatech=EAGER

/**
 * @param Image $image
 * To ensure object initialization, path required in postRemove method
 */
public function preRemove(Image $image)
{
    $image->getPath();
}

/**
 * @param Image $image
 */
public function postRemove(Image $image)
{
    $imagePath = $this->imageHelper->getImagePath($image);

    $this->filesystem->remove($imagePath);
}
 类似资料:
  • 问题内容: 请考虑以下示例Java类(下面的pom.xml): 我写一个FileOutputStream,然后尝试删除该文件, 而不先关闭Stream 。这是我最初的问题,当然是错误的,但它导致了一些奇怪的发现。 在Windows 7上运行主方法时,它将产生以下输出: 为什么第一次调用Files.delete()不会引发异常? 为什么以下对Files.exist()的调用返回false? 为什么无

  • 问题内容: @Entity public class Person { 给定以下类结构,当我尝试将新位置添加到“人的位置”列表中时,它总是导致以下SQL查询: 和 Hibernate(3.5.x / JPA 2)删除给定Person的所有关联记录,然后重新插入所有以前的记录以及新的记录。 我有一个想法,即Location上的equals / hashcode方法可以解决此问题,但是它没有任何改变。

  • 我正在使用ListView。向ListView添加新项很容易,但如何删除? 我通过以下方式构建ListView: Listcell Y的一个实例在setGraphics中设置了一个标签。要从我的观察列表中删除与之关联的字符串,我添加了一个MouseEvent处理程序: 虽然这段代码通过从ObservableList中删除每个单击的项目来工作,但listView的行为很奇怪。每次单击标签时,它都会成

  • 在我的应用程序中,左侧有一个,我根据中的选择更新右侧的窗格。一个非常直接的场景。当选择为空时,我会在窗格中显示类似“请进行选择”的消息,即我也会在 ,以防用户单击某些。侦听器还附加到所选的的,以防用户没有单击某些项目,但所选项目的值因某种原因而更改。用户可以通过单击其中的删除按钮来删除特定项目。在任何时候,用户可以点击更新按钮来更新显示当前选择的标签内容,以防我的处理程序错过了一些事件。 在这个简

  • 这个程序打印00,但是如果我注释掉a.store和b.store,而取消注释a.fetch_add和b.fetch_add,这做了完全相同的事情,即都设置了a=1,b=1的值,我永远不会得到00。 是我错过了什么,还是“00”按标准永远不会出现? 下面打印00。 下图从不打印00 再看看这个,多线程原子a b打印00 for memory_order_refield

  • 我有一个程序可以读取两个文件(“incompleta.txt”和“completa.txt”),并比较两个文件中是否有相同的名称。其工作原理如下: 它将“completa.txt”中的所有名称存储在ArrayList中 问题是我的“lista.txt”看起来像这样:Image 第二条线完全是混乱和扭曲的。 我的Java代码: