为什么我不能直接使用另一个table后面的列不在函数中?
Why I can't directly use the column from another table behind not in function?
我有两张桌子。一个是客户,一个是订单。这是创建两个表的原始数据:
Create table If Not Exists Customers (Id int, Name varchar(255));
Create table If Not Exists Orders (Id int, CustomerId int);
insert into Customers (Id, Name) values ('1', 'Joe');
insert into Customers (Id, Name) values ('2', 'Henry');
insert into Customers (Id, Name) values ('3', 'Sam');
insert into Customers (Id, Name) values ('4', 'Max');
insert into Orders (Id, CustomerId) values ('1', '3');
insert into Orders (Id, CustomerId) values ('2', '1');
现在我必须编写一个 SQL 查询来查找所有从未订购过任何东西的客户。
这是正确答案:
select
name as customer
from
customers
where
customers.id
not in
(select
customerid
from
orders);
这是我的回答:
select
name as customer
from
customers
where
customers.id
not in
orders.customerid;
并且 MySQL 对我的代码的响应是 "Error Code: 1064. 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 'orders.customerid' at line 8"。
我的代码有什么问题?
您的代码缺少对订单 table 的引用。必须在正确的 FROM 和 WHERE 子句或正确的 JOIN 子句
中声明有效 SQL 语法的引用
对于 "not in",您可以使用左连接并检查空值
select name as customer
from customers
left join orders on orders.customerid = customer.id
where orders.customerid is null
如果您不声明对 table 命令的引用,数据库引擎就无法知道必须从哪个 table 检索列值。 table 声明必须明确并在适当的子句(from 或 join)中赋值。
IN
不是函数 BTW,需要 "set" 或 "list" 作为第二个运算符,通常可以通过子查询(或通过给出一堆逗号)获得分隔的文字)。在您的情况下,子查询需要从订单中获取所有客户 ID。所以你可以试试:
SELECT c.name AS customer
FROM customers c
WHERE c.id NOT IN (SELECT o.customerid
FROM orders o);
但不使用 NOT IN
,NOT EXISTS
和相关子查询可能是一个更好的,可能是性能更高的解决方案,特别是如果存在 (orders.customerid)
上的索引:
SELECT c.name AS customer
FROM customers c
WHERE NOT EXISTS (SELECT *
FROM orders o
WHERE o.customerid = c.id);
你问 "why" 关于一种已有将近半个世纪历史的语言。恕我直言,其中一些语言设计决策很难恢复。要获得完美答案,您需要找到 Codd、Chamberlin 和 Boyce。为什么要问为什么?
在更抽象的层面上:SQL 从根本上说是一种集合操作语言。表包含行集。 SELECT 查询从其他行集创建自己的集——它们甚至称为结果集。
在 SQL 世界中,列本身不是一个集合。您可以通过这样说来从列中创建一个集合。
SELECT customerid FROM orders
并且,NOT IN() 操作采用仅包含一列的集合作为其参数。因此,您可以提供 NOT IN() 和 IN() 集,但不能提供列。
我有两张桌子。一个是客户,一个是订单。这是创建两个表的原始数据:
Create table If Not Exists Customers (Id int, Name varchar(255));
Create table If Not Exists Orders (Id int, CustomerId int);
insert into Customers (Id, Name) values ('1', 'Joe');
insert into Customers (Id, Name) values ('2', 'Henry');
insert into Customers (Id, Name) values ('3', 'Sam');
insert into Customers (Id, Name) values ('4', 'Max');
insert into Orders (Id, CustomerId) values ('1', '3');
insert into Orders (Id, CustomerId) values ('2', '1');
现在我必须编写一个 SQL 查询来查找所有从未订购过任何东西的客户。
这是正确答案:
select
name as customer
from
customers
where
customers.id
not in
(select
customerid
from
orders);
这是我的回答:
select
name as customer
from
customers
where
customers.id
not in
orders.customerid;
并且 MySQL 对我的代码的响应是 "Error Code: 1064. 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 'orders.customerid' at line 8"。
我的代码有什么问题?
您的代码缺少对订单 table 的引用。必须在正确的 FROM 和 WHERE 子句或正确的 JOIN 子句
中声明有效 SQL 语法的引用对于 "not in",您可以使用左连接并检查空值
select name as customer
from customers
left join orders on orders.customerid = customer.id
where orders.customerid is null
如果您不声明对 table 命令的引用,数据库引擎就无法知道必须从哪个 table 检索列值。 table 声明必须明确并在适当的子句(from 或 join)中赋值。
IN
不是函数 BTW,需要 "set" 或 "list" 作为第二个运算符,通常可以通过子查询(或通过给出一堆逗号)获得分隔的文字)。在您的情况下,子查询需要从订单中获取所有客户 ID。所以你可以试试:
SELECT c.name AS customer
FROM customers c
WHERE c.id NOT IN (SELECT o.customerid
FROM orders o);
但不使用 NOT IN
,NOT EXISTS
和相关子查询可能是一个更好的,可能是性能更高的解决方案,特别是如果存在 (orders.customerid)
上的索引:
SELECT c.name AS customer
FROM customers c
WHERE NOT EXISTS (SELECT *
FROM orders o
WHERE o.customerid = c.id);
你问 "why" 关于一种已有将近半个世纪历史的语言。恕我直言,其中一些语言设计决策很难恢复。要获得完美答案,您需要找到 Codd、Chamberlin 和 Boyce。为什么要问为什么?
在更抽象的层面上:SQL 从根本上说是一种集合操作语言。表包含行集。 SELECT 查询从其他行集创建自己的集——它们甚至称为结果集。
在 SQL 世界中,列本身不是一个集合。您可以通过这样说来从列中创建一个集合。
SELECT customerid FROM orders
并且,NOT IN() 操作采用仅包含一列的集合作为其参数。因此,您可以提供 NOT IN() 和 IN() 集,但不能提供列。