选择具有相同城市的用户

时间:2018-10-18 18:04:51

标签: mysql sql

我有一个用于多对多关系的用户(标识,名称),城市(标识,名称)和一个联接表users_cities(用户标识,城市标识),例如:

城市

 id | name 
----|------ 
  1 | NY   
  2 | LA   
  3 | CH   
  4 | HU

用户

 id | name 
----|------ 
  1 | James   
  2 | Michael  
  3 | Robert   
  4 | Maria

users_cities

 user_id | city_id 
---------|--------- 
       1 |       1 
       1 |       3 
       2 |       3 
       3 |       1 
       3 |       4 
       4 |       1 
       4 |       3

有一个用于用户索引的策略,当前用户只能看到没有城市的用户列表。另外:

拥有某些城市的用户可以看到没有城市的用户。没有城市的用户可以看到没有城市的用户。

对于给定的用户ID,我该如何选择所有具有相同城市或城市子集的用户?

例如,当用户James(分别位于城市1和3)登录时,我希望他看到:

 id | name 
----|------ 
  1 | James   
  2 | Michael 
  4 | Maria

将用户Robert丢弃,因为Robert的城市为4,而James没有。或者等效地,罗伯特的城市(1,4)并不是詹姆斯(1,3)的子集


EDIT : 此sql显示示例数据和用户James的正确输出:

SELECT users.* 
FROM   users 
       INNER JOIN users_cities 
               ON users_cities.user_id = users.id 
       INNER JOIN cities 
               ON cities.id = users_cities.city_id 
GROUP  BY users.id 
HAVING Group_concat(cities.id) IN ('1,3', '3,1', '1', '3')

See this DB fiddle

并显示正确的输出:

 id | name 
----|------ 
  1 | James   
  2 | Michael
  4 | Maria

但是我不知道如何生成1,3('1,3','3,1','1','3')的所有组合以用于给定用户的hading子句ID


此问题与SELECTING with multiple WHERE conditions on same column有很大不同,因为过滤用户的条件不在同一列或表中,而是在多对多关系中。

1 个答案:

答案 0 :(得分:0)

我会做的:

SELECT users.* 
FROM   users 
       INNER JOIN users_cities ON users_cities.user_id = users.id 
WHERE  users_cities.id in (select id from users_cities.user_id = <JamesId>)
       and users.id <> <JamesId>;

这里有技巧:

select id, name
from (
SELECT users.*, users_cities.city_id, ifnull(james_cities.city_id, 'NULL') james_city_id
FROM users
INNER JOIN users_cities ON users_cities.user_id = users.id
left join users_cities james_cities on james_cities.city_id = users_cities.city_id and james_cities.user_id = 1
union
select users.*, 'NULL', 'NULL'
from users where id not in (select user_id from users_cities)
) a 
group by id, name
having group_concat(city_id) = group_concat(james_city_id)