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 中不存在,所以始终为真
我最近偶然发现了这个例子。为什么在临时 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 中不存在,所以始终为真