为什么我不能直接使用另一个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 INNOT 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() 集,但不能提供列。