MySQL 中 DELETE 语句中可以使用别名么?

某天,正按照业务的要求删除不需要的数据,在执行 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 开始单表多表都可以使用别名。

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

关于 SQLE

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

SQLE 获取

类型地址
版本库https://github.com/actiontech/sqle
文档https://actiontech.github.io/sqle-docs/
发布信息https://github.com/actiontech/sqle/releases
数据审核插件开发文档https://actiontech.github.io/sqle-docs/docs/dev-manual/plugins/howtouse
### 关于DELETE语句表别名使用限制及其解决方法 在MySQL中,DELETE语句支持表别名的功能,但在实际应用中有一定的语法约束。以下是关于为何DELETE语句不能直接使用表别名的原因分析以及如何正确编写的解决方案。 #### 1. DELETE语句中的表别名限制原因 MySQL的设计原则决定了其对DELETE语句的支持方式较为严格。为了防止歧义并保持一致性,在涉及复杂查询(如多表操作或子查询)时,MySQL要求开发者显式指定要删除的目标表[^1]。因此,当尝试仅通过别名执行DELETE操作而未遵循特定语法结构时,会引发语法错误。 具体来说,如果只简单地写出类似于`DELETE FROM table_alias WHERE condition`这样的形式,则可能导致解析器无法识别真正的目标实体是什么——即到底是应该作用到原始基础表还是仅仅针对当前上下文中定义的那个临时标记对象上去做改动处理动作而已[^2]。 #### 2. 正确书写带表别名DELETE语句 为了避免上述提到的问题发生,并能够灵活运用表别名来简化复杂的SQL表达逻辑或者提高可读性等方面考虑的话,可以按照下面给出的标准模式来进行构建相应的命令字符串: - **标准格式**: ```sql DELETE alias_name FROM actual_table_name AS alias_name WHERE conditions; ``` 这里的关键点在于必须先声明所使用的真正物理存储单元的名字(`actual_table_name`)紧接着才是赋予它的另一个易于理解的小标签(alias),最后再上任何必要的过滤条件部分完成整个句子构成过程[^3]。 例如,假设存在一张名为`orders`的数据集合体,并希望基于订单编号字段(order_id)等于某个固定值(比如"123")的前提下来清除掉符合条件记录项的操作需求下,那么对应的实现代码将会像这样呈现出来: ```sql DELETE t FROM orders AS t WHERE t.order_id = "123"; ``` 此版本不仅满足了基本功能诉求同时也兼顾到了良好的编码习惯展示效果[^4]^. 另外值得注意的是,在某些数据库管理系统(DBMS)里可能还存在着其他额外的要求或者是差异化的表现行为特征等问题需要注意规避风险情况的发生几率提升整体稳定性水平度量指标数值大小范围内的合理区间之内[^5]. ### 总结 综上所述,虽然MySQL允许我们在DELETE语句使用表别名以增强灵活性和清晰度,但我们仍需遵守严格的语法规则才能成功运行此类查询。这通常意味着我们需要明确指出哪个真实存在的表格是我们打算修改的对象,然后再为其分配一个更简洁易记的新名字作为替代品供后续引用之便。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值