MySql 使用递归函数时遇到的级联删除问题

以下两段SQL的写法看似相同,结果效果却是不同的

写法A

DELETE OM_ORGANIZATION,
OM_POSITION
FROM
OM_ORGANIZATION
LEFT JOIN OM_POSITION ON OM_POSITION.ORG_ID = OM_ORGANIZATION.ORG_ID
WHERE
FIND_IN_SET(
          OM_ORGANIZATION.ORG_ID,
          OM_ORGANIZATION_QUERY_CHILDREN (‘52037b7c-0f01-41f6-849f-4f99ad8f2422‘)
)

这个写法是不正确的,先来介绍一下这个

OM_ORGANIZATION_QUERY_CHILDREN 

函数

BEGIN
DECLARE sTemp VARCHAR(4000);
DECLARE sTempChd VARCHAR(4000);

SET sTemp = ‘$‘;
SET sTempChd = id;

WHILE sTempChd is not NULL DO
SET sTemp = CONCAT(sTemp,‘,‘,sTempChd);
SELECT group_concat(ORG_ID) INTO sTempChd FROM OM_ORGANIZATION where FIND_IN_SET(PARENT_ORG_ID,sTempChd)>0;
END WHILE;
return sTemp;
END

id(VARCHAR(40))是这个函数的参数

这个函数的执行结果是返回一个拼起来的字符串,字符串根据主键ORG_ID、父节点主键PARENT_ORG_ID两个字段,递归查询出OM_ORGANIZATION中所有以传入参数作为根节点的主键,然后拼成一个类似 【1,2,3,4】的可供FIND_IN_SET使用的字符串(这里定义成VARCHAR(4000))。

写法A中:每删除一条数据都有可能造成OM_ORGANIZATION_QUERY_CHILDREN (‘1‘)这个函数的执行结果变化,

假设删除第一条的时候,这个函数的执行结果是【1,2,3】,其中’2‘的父节点是’1‘,’3‘的父节点也是’1‘,那么在删除完’2‘之后,函数的结果变成了【1,3】,而后边与’2‘存在关系的OM_POSITION就无法被删掉了。

正确的写法应该如下:

写法B:

DELETE OM_ORGANIZATION,
 OM_POSITION
FROM
    (
        SELECT
            ORG_ID
        FROM
            OM_ORGANIZATION
        WHERE
            FIND_IN_SET(
          OM_ORGANIZATION.ORG_ID,
          OM_ORGANIZATION_QUERY_CHILDREN (‘52037b7c-0f01-41f6-849f-4f99ad8f2422‘)
            )
    ) T
INNER JOIN OM_ORGANIZATION ON T.ORG_ID = OM_ORGANIZATION.ORG_ID
LEFT JOIN OM_POSITION ON OM_POSITION.ORG_ID = OM_ORGANIZATION.ORG_ID

这样,函数只执行一遍,其结果将一直保持不变,直到SQL结束。不仅提高了效率,还避免了错误。

时间: 2024-10-31 16:35:28

MySql 使用递归函数时遇到的级联删除问题的相关文章

[原创]MYSQL中利用外键实现级联删除和更新

MySQL中利用外键实现级联删除.更新 MySQL支持外键的存储引擎只有InnoDB,在创建外键的时候,要求父表必须有对应的索引,子表在创建外键的时候也会自动创建对应的索引.在创建索引的时候,可以指定在删除.更新父表时,对子表进行的相应操作,包括RESTRICT.NO ACTION.SET NULL和CASCADE.其中RESTRICT和NO ACTION相同,是指在子表有关联记录的情况下父表不能更新:CASCADE表示父表在更新或者删除时,更新或者删除子表对应记录:SET NULL则是表示父表

MySQL中利用外键实现级联删除、更新

MySQL支持外键的存储引擎只有InnoDB,在创建外键的时候,要求父表必须有对应的索引,子表在创建外键的时候也会自动创建对应的索引.在创建索引的时候,可以指定在删除.更新父表时,对子表进行的相应操作,包括RESTRICT.NO ACTION.SET NULL和CASCADE.其中RESTRICT和NO ACTION相同,是指在子表有关联记录的情况下父表不能更新:CASCADE表示父表在更新或者删除时,更新或者删除子表对应记录:SET NULL则是表示父表在更新或者删除的时候,子表的对应字段被S

mysql级联删除更新

首先,目前在产品环境可用的MySQL版本(指4.0.x和4.1.x)中,只有InnoDB引擎才允许使用外键,所以,我们的数据表必须使用InnoDB引擎. 下面,我们先创建以下测试用数据库表: CREATE TABLE `roottb` (   `id` INT(11) UNSIGNED AUTO_INCREMENT NOT NULL,   `data` VARCHAR(100) NOT NULL DEFAULT '',   PRIMARY KEY (`id`) ) TYPE=InnoDB; CR

关于数据的级联删除和更新

两张表:“ProductCategory”,“Product”. 有一个需求是这样的:在删除某个ProductCategory 的时候,同时删除该Category的products. 在MySQL中利用外键实现级联删除 创建以下测试用数据库表: CREATE TABLE `roottb` ( `id` INT(11) UNSIGNED AUTO_INCREMENT NOT NULL, `data` VARCHAR(100) NOT NULL DEFAULT '', PRIMARY KEY (`i

【Python】Django数据模型、级联删除、级联更新、ER图导出等

在本文中,我们将向读者详细介绍如何在更新和删除父表数据的同时,触发有关子表数据的级联更新和删除操作.您将看到当使用InnoDB表的时候,借助于外键约束就可以轻松搞定这一过程. 一.利用外键约束更新并删除MySQL中的数据 我们知道,开发能够维护多个表的完整性的数据库驱动的应用程序是一件非常复杂的事情--即使应用程序所面对的是当前最流行的开源关系型数据库管理系统MySQL服务器时也不例外.如果一个应用程序必须处理多个数据库表,而这些表之间有存在着某些预定义的关系,这时一旦父表中的数据被更新或者删除

关于mysql的级联删除(之前好多人咨询过我)

最近许多项目咨询,mysql进行级联删除的问题.在此做个系统的扫盲. 一 什么是级联删除. 即依赖于外键关系,删除父表时,一并删除独立依赖于此表的子表. 二 如何做到级联删除 其实非常简单,需要建立表的时候明确指定,外键依赖关系的属性为允许级联删除. ON DELETE CASCADE 三 使用级联删除有什么优缺点 优点很明显,程序删除时变简单了. 缺点也很明显,业务设计时做删除考虑时需要更高的要求.否则易出问题.

MySQL外键及级联删除 && 表的存储引擎与创建索引 && 删除数据库和表

Messages表: mysql>create table Messages( ->message_id int auto_increment primary key, ->user_name varchar(50) not null, ->author_id int not null, ->body text, ->forum_id int not null); Forums表: mysql>create table Forums( ->forum_id

多表联查时的sql删除语句的写法,即级联删除,将相关联的数据级联删除

mysql多表关联删除 DELETE删除多表数据,怎样才能同时删除多个关联表的数据呢?这里做了深入的解释: 1 delete from t1 where 条件 2 delete t1 from t1 where 条件 3 delete t1 from t1,t2 where 条件 4 delete t1,t2 from t1,t2 where 条件 前 3者是可行的,第4者不可行. 也就是简单用delete语句无法进行多表删除数据操作,不过可以建立级联删除,在两个表之间建立级联删除关系,则可以实

MySql 修改外键 支持级联删除

首先必须要有外键,InnoDB甚么的都不说了,直接上修改句子. ALTER TABLE `t_terminal` DROP FOREIGN KEY `FK704405E7F06A14EF`; ALTER TABLE `t_terminal` ADD CONSTRAINT `FK704405E7F06A14EF` FOREIGN KEY (`userId`) REFERENCES `t_user` (`id`) ON DELETE CASCADE ON UPDATE RESTRICT; 要先删除该