views:

44

answers:

2

Hi everyone,

I have a many-to-many relationship, in which I query all M which have a specific N, for example:

SELECT M.* FROM M INNER JOIN ManyToManyTable
ON M.Id = ManyToManyTable.M
WHERE ManyToManyTable.N = @Id

Or:

SELECT M.* FROM M INNER JOIN ManyToManyTable
ON M.Id = ManyToManyTable.M
INNER JOIN N
ON N.Id = ManyToManyTable.N
WHERE N.Id = @Id

My question is, how to efficiently inverse the query so I get all M which do not have a specific N?

That is, like a NOT IN the selects above, but without the NOT IN statement, if possible.

+1  A: 

Quick and dirty:

SELECT M.* FROM M
WHERE M.id NOT IN
    (SELECT M.id FROM M INNER JOIN ManyToManyTable
     ON M.Id = ManyToManyTable.M
     WHERE ManyToManyTable.N = @Id)

Better:

SELECT M.*
FROM M LEFT JOIN ManyToManyTable
ON M.Id = ManyToManyTable.M
WHERE ManyToManyTable.M IS NULL
Anax
+5  A: 

In SQL Server 'NOT EXISTS' is generally more efficient than the OUTER JOIN approach.

SELECT M.* FROM M  
WHERE NOT EXISTS
 (SELECT * FROM ManyToManyTable MMT
           WHERE MMT.M = M.Id AND N=@Id )
Martin Smith
+1. Just what I was about to suggest.
Brian Hooper
Most useful answer.
Anax
Took me a while to fit my exact needs (that is-- translate to NHibernate), but works beautifully! Great answer!
GeReV