mysql - Zend Framework - 数据库表递归级联删除问题

标签 mysql zend-framework zend-db-table referential-integrity mysql-error-1451

我的情况可能有点不正常,但我在我的 MySQL 数据库中定义了外键,同时在 Zend_Db_Table 类中强制引用完整性。这些表使用 InnoDB 存储引擎。

删除记录时,Zend Framework 会通过表模型中的$_referenceMap 正确识别直接子项并删除它们。但是,如果有直系子项的任何子项,我会从数据库返回有关违反该外键参照完整性的错误:SQLSTATE[23000]: Integrity constraint violation: 1451 Cannot delete or update a parent行:外键约束失败。看起来 Zend_Db_Table_Abstract 没有以递归方式强制引用完整性。

有没有人遇到过这个?是Zend Framework bug吗?解决方法?修复?


更新

将近一个星期后,我还没有对这个问题作出答复。我想我必须自己扩展 Zend_Db_Table_Row_Abstract 类才能完成此任务。

最佳答案

我最终扩展了 Zend_Db_Table_Abstract 类来实现这一点。 _cascadeDelete 公共(public)函数调用数据库适配器delete 函数。我进行了更改,以便调用 Zend_Db_Table_Row_Abstract 中的 delete 函数。这使得记录删除具有递归性。

更新:我添加了删除时设置为 self::SET_NULL 的代码。

这是我对 _cascadeDelete 的修改版本:

/**
 * Called by parent table's class during delete() method.
 *
 * @param  string $parentTableClassname
 * @param  array  $primaryKey
 * @return int    Number of affected rows
 */
public function _cascadeDelete($parentTableClassname, array $primaryKey)
{
    $this->_setupMetadata();
    $rowsAffected = 0;
    foreach ($this->_getReferenceMapNormalized() as $map) {
        if ($map[self::REF_TABLE_CLASS] == $parentTableClassname && isset($map[self::ON_DELETE])) {
            switch ($map[self::ON_DELETE]) {
                case self::CASCADE:
                    $select = $this->select();
                    for ($i = 0; $i < count($map[self::COLUMNS]); ++$i) {
                        $col = $this->_db->foldCase($map[self::COLUMNS][$i]);
                        $refCol = $this->_db->foldCase($map[self::REF_COLUMNS][$i]);
                        $type = $this->_metadata[$col]['DATA_TYPE'];
                        $select->where($this->_db->quoteIdentifier($col, true) . ' = ?',
                            $primaryKey[$refCol], $type);
                    }
                    $rows = $this->fetchAll($select);
                    $rowsAffected += count($rows);
                    foreach ($rows as $row) {
                        $row->delete();
                    }
                    break;
                case self::SET_NULL: {
                    $update = array();
                    $where = array();
                    for ($i = 0; $i < count($map[self::COLUMNS]); ++$i) {
                        $col = $this->_db->foldCase($map[self::COLUMNS][$i]);
                        $refCol = $this->_db->foldCase($map[self::REF_COLUMNS][$i]);
                        $type = $this->_metadata[$col]['DATA_TYPE'];
                        $update[$col] = null;
                        $where[] = $this->_db->quoteInto(
                            $this->_db->quoteIdentifier($col, true) . ' = ?',
                            $primaryKey[$refCol], $type);
                    }
                    $rowsAffected += $this->update($update, $where);
                    break;
                }
                default:
                    // no action
                    break;
            }
        }
    }
    return $rowsAffected;
}

关于mysql - Zend Framework - 数据库表递归级联删除问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3023809/

相关文章:

python - 无法在 Windows 上安装正确版本的 mysql(8.0.13)

php - Wamp - 新安装后我无法登录 phpmyadmin #2002

php - bootstrap 通常如何工作,尤其是在 Zend Framework 中?

php - ZendDb中类似mysql_affected_rows()的函数是什么

php - Zend_Db 子查询

mysql - 如果不存在则插入,否则只需在 mysql 中选择

php - 如何在 Zend_Form 中添加 '[]' 到表单元素名称?

php - 如何在 Zend Framework 中使用 Join

zend-framework - 将查询转换为使用 Zend_Db_Select

php - 如何将 Codeigniter 预格式化查询转换为原始 Mysql 查询(例如 $this->db->select ('table1.*,table2.*,table3.*' ))