MSSQL temp table returns 使用 IN 子查询时的所有行

MSSQL temp table returns all rows when using IN subquery

我最近偶然发现了这个例子。为什么在临时 table "id" 上使用未定义的列时查询 return 所有行。这仅适用于 id,并且仅在使用 IN 子查询时有效。

DECLARE @test TABLE
( Id INT)

INSERT INTO @test
SELECT 1 UNION ALL
SELECT 2 UNION ALL
SELECT 3

SELECT * FROM @test

SELECT 1 AS 'myid'
INTO #tmpId

--Returns one rows - OK
SELECT * FROM @test WHERE id in (SELECT myId FROM #tmpId)
--Returns all rowns - WHY
SELECT * FROM @test WHERE id in (SELECT id FROM #tmpId)
--ERROR - no column - OK
SELECT * FROM @test WHERE id in (SELECT asdf FROM #tmpId)
--ERROR - no column - OK
SELECT id FROM #tmpId

DROP TABLE #tmpId

当您有多个 table 时,总是 使用 table 别名。第一个查询应该写成:

SELECT t.* FROM @test t WHERE t.id in (SELECT tt.myid FROM #tmpId tt)

这就是你想要的。

如果将第二个查询写成:

SELECT t.* FROM @test t WHERE t.id in (SELECT tt.id FROM #tmpId tt)

你会得到一个错误。您的版本被解释为相关子查询:

SELECT t.* FROM @test t WHERE t.id in (SELECT t.id FROM #tmpId tt)

那不是你想要的。相当于:

SELECT t.* FROM @test t WHERE t.id IS NOT NULL;

在第二个查询中,您将 Id 列与其自身进行比较,因为内部 table 中不存在,所以始终为真