在MySQL查询的WHERE子句中使用列别名会产生错误

Jam*_*mes 183 mysql sql mysql-error-1054

我正在运行的查询如下,但是我收到此错误:

#1054 - 'IN/ALL/ANY子查询'中的未知列'guaranteed_postcode'

SELECT `users`.`first_name`, `users`.`last_name`, `users`.`email`,
SUBSTRING(`locations`.`raw`,-6,4) AS `guaranteed_postcode`
FROM `users` LEFT OUTER JOIN `locations`
ON `users`.`id` = `locations`.`user_id`
WHERE `guaranteed_postcode` NOT IN #this is where the fake col is being used
(
 SELECT `postcode` FROM `postcodes` WHERE `region` IN
 (
  'australia'
 )
)
Run Code Online (Sandbox Code Playgroud)

我的问题是:为什么我无法在同一个数据库查询的where子句中使用假列?

vic*_*ugo 410

您只能在GROUP BY,ORDER BY或HAVING子句中使用列别名.

标准SQL不允许您引用WHERE子句中的列别名.施加此限制是因为当执行WHERE代码时,可能尚未确定列值.

MySQL文档复制

正如评论中所指出的那样,使用HAVING可以完成工作.尽管如此,请务必阅读此WHERE与HAVING.

  • 如果其他人有一个与我相同的问题,即在where子句中使用别名col失败 - 将'WHERE'交换为'HAVING'直接修复它+1一个好的答案. (37认同)
  • 在你的情况下,这可能是重要的,也可能不重要,但是`HAVING`的执行速度比`WHERE'慢 (9认同)
  • `have` 工作的原因是因为列值必须在您到达 `have` 时计算。如上所述,`where` 不是这种情况。 (3认同)

rod*_*ion 24

正如Victor指出的那样,问题在于别名.但是,通过将表达式直接放入WHERE x IN y子句中,可以避免这种情况:

SELECT `users`.`first_name`,`users`.`last_name`,`users`.`email`,SUBSTRING(`locations`.`raw`,-6,4) AS `guaranteed_postcode`
FROM `users` LEFT OUTER JOIN `locations`
ON `users`.`id` = `locations`.`user_id`
WHERE SUBSTRING(`locations`.`raw`,-6,4) NOT IN #this is where the fake col is being used
(
 SELECT `postcode` FROM `postcodes` WHERE `region` IN
 (
  'australia'
 )
)
Run Code Online (Sandbox Code Playgroud)

但是,我认为这是非常低效的,因为子查询必须为外部查询的每一行执行.

  • @rodion,是的,我相信这是**严重**缓慢且低效的。 (2认同)

Jon*_*oni 20

标准SQL(或MySQL)不允许在WHERE子句中使用列别名,因为

在评估WHERE子句时,可能尚未确定列值.

(来自MySQL文档).您可以做的是计算WHERE子句中的列值,将值保存在变量中,并在字段列表中使用它.例如,你可以这样做:

SELECT `users`.`first_name`, `users`.`last_name`, `users`.`email`,
@postcode AS `guaranteed_postcode`
FROM `users` LEFT OUTER JOIN `locations`
ON `users`.`id` = `locations`.`user_id`
WHERE (@postcode := SUBSTRING(`locations`.`raw`,-6,4)) NOT IN
(
 SELECT `postcode` FROM `postcodes` WHERE `region` IN
 (
  'australia'
 )
)
Run Code Online (Sandbox Code Playgroud)

这避免了在表达式变得复杂时重复表达式,使代码更易于维护.

  • 这不会与文档[说]相矛盾(http://stackoverflow.com/questions/16715504/mysql-define-a-variable-within-select-and-use-it-within-the-same-select/16715618#16715618)*"作为一般规则,您不应该为用户变量赋值并在同一语句中读取值.您可能会得到您期望的结果,但这不能保证."*? (7认同)

小智 14

也许我的答案为时已晚,但这可以帮助别人.

您可以使用另一个select语句将其括起来并使用where子句.

SELECT * FROM (Select col1, col2,...) as t WHERE t.calcAlias > 0
Run Code Online (Sandbox Code Playgroud)

calcAlias是计算的别名列.


Het*_*ett 8

您可以对SELECT字段和别名中计算的过滤器使用HAVING子句