Using SQL Server Management Studio, I am getting some undesired results (looks like a bug to me..?)
If I use (FIELD rather than field for the other_table):
SELECT * FROM main_table WHERE field IN (SELECT FIELD FROM other_table)
I get all results from main_table.
Using the correct case:
SELECT * FROM main_table WHERE field IN (SELECT field FROM other_table)
I get the expected results where field appears in other.
Running the subquery on it’s own:
SELECT FIELD FROM other_table
I get an invalid column name error.
Surely I should get this error in the first case?
Is this related to collation?
The DB is binary collation.
The server is case insensitive however.
It seems to me like the server component is saying “this code is OK” and not allowing the DB to say the field is the wrong name..?
What are my options for a solution?
Let’s illustrate what is happening using something that doesn’t depend on case sensitivity:
Results:
Why doesn’t that raise an error? SQL Server is looking at your queries and seeing that the column1 inside can’t possibly be in other_table, so it is extrapolating and "using" the column1 that exists in the outer referenced table (just like you could reference a column that only exists in the outer table without a table reference). Think about this variation:
Results:
Again SQL Server knows that column1 in the where clause also doesn’t exist in the locally referenced table, but it tries to find it in the outer scope. So in an imaginary world you might consider the query to actually be saying:
(Which is not how I typed it, but if I do type it that way, it still works.)
It doesn’t make logical sense in some of the cases but this is the way the query engine does it and the rule has to be applied consistently. In your case (no pun intended), you have an extra complication: case sensitivity. SQL Server didn’t find
FIELDin your subquery, but it did find it in the outer query. So a couple of lessons: