经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 数据库/运维 » MySQL » 查看文章
解决MySQL报错:You?can‘t?specify?target?table?‘region‘?for?update?in?FROM?clause
来源:jb51  时间:2023/2/2 9:09:15  对本文有异议

前言

首先明确一点这个错误只会发生在delete语句或者update语句,拿update来举例 : update A表 set A列 = (select B列 from A表); 这种写法就会报这个错误,原因:你又要修改A表,然后又要从A表查数据,而且还是同层级。Mysql就会认为是语法错误!

嵌套一层就可以解决,update A表 set A列 = (select a.B列 from (select * from A表) a); 当然这个只是个示例,这个示例也存在一定的问题,比如(select a.B列 from (select * from A表) a)他会查出来多条,然后赋值的时候会报 1242 - Subquery returns more than 1 row

嵌套一层他就可以和update撇清关系,会优先查括号里面的内容,查询结果出来过后会给存起来,类似临时表,可能有的人该好奇了,update A表 set A列 = (select B列 from A表); 我明明加括号了呀,难道不算嵌套吗,当然不算,那个括号根本没有解决他们之间的层次关系!

所谓层次关系就是一条sql当中谁先执行谁后执行,能理解层次关系的尽量要理解,不懂也没关系,下面我提供了四个案例 供大家参考!

示例一

以这张表为例:

  1. DROP TABLE IF EXISTS `region`;
  2. CREATE TABLE `region`
  3. (
  4. `Id` int(11) NOT NULL COMMENT '主键id',
  5. `Name` varchar(40) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL COMMENT '名称',
  6. `Pid` int(11) NULL DEFAULT NULL COMMENT '父类id',
  7. `status` varchar(2) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT '1' COMMENT '1:启用,2:禁用',
  8. PRIMARY KEY (`Id`) USING BTREE,
  9. INDEX `FK_CHINA_REFERENCE_CHINA`(`Pid`) USING BTREE
  10. ) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '省市区表' ROW_FORMAT = DYNAMIC;

错误用法:

这个例子就是典型,我要修改region表数据,要将pid 赋值为region当中的某一列数据,这种写法就会报错!mysql不支持 同一张表当中 既要修改又要查询

  1. UPDATE region SET pid = (select pid FROM region WHERE NAME = '市辖区') where name='北京';

明确一点,这种的不是同一张表是不会报错的!

  1. UPDATE region SET pid = (select id FROM banner) ;

正确用法:

  1. UPDATE region
  2. SET pid = ( SELECT a.pid FROM ( SELECT Pid FROM region WHERE NAME = '市辖区' ) a )
  3. WHERE
  4. NAME = '北京';

示例二

错误用法:

  1. UPDATE region a
  2. SET Name = '1'
  3. WHERE
  4. a.pid IN ( SELECT id FROM region WHERE NAME = '市辖区' );

正确用法:

  1. UPDATE region a
  2. SET Name = '1'
  3. WHERE
  4. a.pid IN ( SELECT b.id FROM (select * FROM region) b WHERE b.NAME = '市辖区' );

示例三

这个sql没有错误示例,只有正确示例,相对来说比较复杂点!

  1. UPDATE region a
  2. SET pid = (
  3. SELECT
  4. b.pid
  5. FROM
  6. ( SELECT id, pid FROM region b WHERE b.NAME = '市辖区' ) b
  7. WHERE
  8. a.Pid = b.id
  9. )
  10. WHERE
  11. a.pid IN ( SELECT c.id FROM ( SELECT * FROM region ) c WHERE c.NAME = '市辖区' );

示例四

再来看一个删除的示例

错误用法:

  1. delete from tbl where id in
  2. (
  3. select max(id) from tbl a where EXISTS
  4. (
  5. select 1 from tbl b where a.tac=b.tac group by tac HAVING count(1)>1
  6. )
  7. group by tac
  8. );

正确用法:

  1. delete from tbl where id in
  2. (
  3. select a.id from
  4. (
  5. select max(id) id from tbl a where EXISTS
  6. (
  7. select 1 from tbl b where a.tac=b.tac group by tac HAVING count(1)>1
  8. )
  9. group by tac
  10. ) a
  11. );

需要注意的地方

(select...) 一定要加个别名,例如:(select...) a否则报错如下:

总结

到此这篇关于解决MySQL报错:You can‘t specify target table ‘region‘ for update in FROM clause的文章就介绍到这了,更多相关MySQL报错解决内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号