我有三个表:页面,附件,页面附件
我有这样的数据:
page
ID NAME
1 first page
2 second page
3 third page
4 fourth page
attachment
ID NAME
1 foo.word
2 test.xsl
3 mm.ppt
page-attachment
ID PAGE-ID ATTACHMENT-ID
1 2 1
2 2 2
3 3 3
我希望每个页面的附件数量当该数字为0 时。我尝试过:
select page.name, count(page-attachment.id) as attachmentsnumber
from page
inner join page-attachment on page.id=page-id
group by page.id
我收到了这个输出:
NAME ATTACHMENTSNUMBER
second page 2
third page 1
我想得到这个输出:
NAME ATTACHMENTSNUMBER
first page 0
second page 2
third page 1
fourth page 0
如何获得0部分?
答案 0 :(得分:27)
将“内部联接”更改为“左外部联接”,这意味着“即使右侧没有匹配的行,也可以”获取联接左侧的所有行。“
select page.name, count(page-attachment.id) as attachmentsnumber
from page
left outer join page-attachment on page.id=page-id
group by page.name
答案 1 :(得分:8)
这是使用子查询的另一种解决方案。
SELECT
p.name,
(
SELECT COUNT(*) FROM [page-attachment] pa
WHERE pa.[PAGE-ID] = p.id
) as attachmentsnumber
FROM page p
答案 2 :(得分:3)
根据数据库的不同,对于速度,您可以使用UNION命令。
SQL更长,但是,根据数据库的不同,它可以通过分离“计算存在的东西”和“计算不存在的东西”来加快速度。
(
select page.name, count(page-attachment.id) as attachmentsnumber
from page
inner join page-attachment on page.id=page-id
group by page.id
)
UNION
(
select page.name, 0 as attachmentsnumber
from page
where page.id not in (
select page-id from page-attachment)
)
我需要此解决方案的数据库在超过一百万个附件中有20页。 UNION让它在13秒内运行而不是很长时间我感到无聊并尝试了另一种方式(在我杀死外连接和子查询方法之前60秒以上)。
答案 3 :(得分:1)
您需要左连接而不是内连接,因为它允许记录不存在。
答案 4 :(得分:1)
LEFT join是你的朋友。 要了解有关不同联接类型的详细信息,请参阅http://en.wikipedia.org/wiki/Join_(SQL)
答案 5 :(得分:0)
使用此:
SELECT p.name,(
SELECT COUNT(*) FROM [page-attachment] pa WHERE pa.[PAGE-ID] = p.id) as attachmentsnumber
FROM page p