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

我正在运行的查询如下,但是我得到这个错误:

#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' ) ) 

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

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

标准SQL不允许引用WHERE子句中的列别名。 这个限制是因为执行WHERE代码时,列值可能还没有确定。

从MySQL文档复制

正如评论中所指出的那样,使用HAVING可以完成这项工作。 请务必阅读此WHERE VS HAVING 。

正如维克多指出的那样,问题在于别名。 这可以通过将表达式直接放入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' ) ) 

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

标准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' ) ) 

这避免了在表达复杂时重复表达,使代码更容易维护。

也许我的回答太晚了,但这可以帮助别人。

你可以用另一个select语句把它括起来,并使用where子句。

 SELECT * FROM (Select col1, col2,...) as t WHERE t.calcAlias > 0 

calcAlias是计算的别名列。

您可以在SELECT字段和别名中使用HAVING子句进行计算

我正在使用MySQL 5.5.24和下面的代码工作:

 select * from ( 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` ) as a WHERE guaranteed_postcode NOT IN --this is where the fake col is being used ( SELECT `postcode` FROM `postcodes` WHERE `region` IN ( 'australia' ) ) 

标准SQL不允许引用WHERE子句中的列别名。 这个限制是强加的,因为当WHERE子句被评估时,列值可能还没有被确定。 例如,以下查询是非法的:

SELECT id,COUNT(*)AS cnt FROM tbl_name WHERE cnt> 0 GROUP BY id;