某天,正依照事务的要求删去不需要的数据,在履行 DELETE 句子时,居然呈现了报错!

作者:林靖华,开源数据库技能爱好者,拿手MySQL和Redis的运维

爱可生开源社区出品,原创内容未经授权不得随意运用,转载请联络小编并注明来历。

本文约 650 字,预计阅读需要 2 分钟。

背景

某天,正依照事务的要求删去不需要的数据,在履行 DELETE 句子时,居然呈现了报错(MySQL 数据库版别 5.7.34):

mysql> delete from test1 t1 where not exists (select 1 from test2 t2 where t1.id=t2.id);
ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 't1 where not exists (select 1 from test2 t2 where t1.id=t2.id)' at line 1

这就有点奇怪了,由于我在履行删去句子之前,履行过同样条件的 SELECT 句子,仅仅把其中的 select * 换成了 delete 罢了,究竟这个语法的报错一般来说原因很大可能是 要害字拼写错误 或者 存在中文符号

排除了上面的原因后,再从句子本身的逻辑来排查,莫非说 DELETE 句子不支撑 not exists 这种写法?好像之前也没听说过这个约束。咱们还是以语法错误这个原由于起点,去查查官方文档看下能不能找出答案。

分析

DELETE 的语法如下:

5.7 单表删去格局

DELETE [LOW_PRIORITY] [QUICK] [IGNORE] FROM tbl_name
    [PARTITION (partition_name [, partition_name] ...)]
    [WHERE where_condition]
    [ORDER BY ...]
    [LIMIT row_count]

仔细比照了以下,发现了一些端倪,这里的语法并没有写出表名的别号用法,莫非是运用了别号的原因?

mysql> delete from test1 where not exists (select 1 from test2 where test1.id=test2.id);
Query OK, 1 row affected (0.00 sec)

经测试去掉了别号还真的履行成功了,但我印象中之前删去数据的时候用过别号,所以我再继续深挖文档查检查。

比照不同地方和不同版别的格局差异后,我终于明白了问题的原因。在不同版别,乃至不同情况下都有差异。

8.0 单表删去格局

DELETE [LOW_PRIORITY] [QUICK] [IGNORE] FROM tbl_name [[AS] tbl_alias]
    [PARTITION (partition_name [, partition_name] ...)]
    [WHERE where_condition]
    [ORDER BY ...]
    [LIMIT row_count]

5.7 和 8.0 多表删去格局

DELETE [LOW_PRIORITY] [QUICK] [IGNORE]
    tbl_name[.*] [, tbl_name[.*]] ...
    FROM table_references
    [WHERE where_condition]
DELETE [LOW_PRIORITY] [QUICK] [IGNORE]
    FROM tbl_name[.*] [, tbl_name[.*]] ...
    USING table_references
    [WHERE where_condition]

经过上面语法比照的不同发现,5.7 的单表删去确实不支撑别号的运用,可是多表删去却支撑(table_references 里包含别号的运用)。

并且在 8.0.16 开端,单表删去已经支撑运用别号了。

For consistency with the SQL standard and other RDBMS, table aliases are now supported in single-table as well as multi-table DELETE statements. (Bug #27455809)

定论

  • MySQL 5.7 运用单表删去句子时,不能运用别号,多表删去可以运用别号。
  • MySQL 8.0.16 开端单表多表都可以运用别号。

更多技能文章,请访问:opensource.actionsky.com/

关于 SQLE

SQLE 是一款全方位的 SQL 质量管理平台,覆盖开发至生产环境的 SQL 审核和管理。支撑主流的开源、商业、国产数据库,为开发和运维提供流程自动化才能,提升上线效率,进步数据质量。