Query SQL, conteggio con conteggio 0
Domanda
Ho tre tabelle:pagina, allegato, pagina-allegato
Ho dati come questi:
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
Vorrei ottenere il numero di allegati per pagina anche quando quel numero è 0.Ho provato con:
select page.name, count(page-attachment.id) as attachmentsnumber
from page
inner join page-attachment on page.id=page-id
group by page.id
ottengo questo output:
NAME ATTACHMENTSNUMBER
second page 2
third page 1
Vorrei ottenere questo output:
NAME ATTACHMENTSNUMBER
first page 0
second page 2
third page 1
fourth page 0
Come ottengo la parte 0?
Soluzione
Cambia il tuo "inner join" in "left external join", che significa "procurami tutte le righe a sinistra del join, anche se non c'è una riga corrispondente a destra".
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
Altri suggerimenti
Ecco un'altra soluzione che utilizza le sottoquery.
SELECT
p.name,
(
SELECT COUNT(*) FROM [page-attachment] pa
WHERE pa.[PAGE-ID] = p.id
) as attachmentsnumber
FROM page p
A seconda del database, per velocità, potresti usare il comando UNION.
L'SQL è più lungo, ma, a seconda del database, accelera le cose separando "conta cose che ci sono" e "conta cose che non ci sono".
(
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)
)
Il database per cui ho bisogno di questa soluzione ha 20 pagine in più di un milione di allegati.UNION l'ha fatto funzionare in 13 secondi invece che così a lungo che mi sono annoiato e ho provato un altro modo (da qualche parte sopra i 60 secondi prima di uccidere i metodi di join esterno e subquery).
Vuoi un join sinistro, invece di un join interno, poiché ciò consente ai record di non esistere.
SINISTRA l'iscrizione è tua amica.Per ulteriori informazioni sui diversi tipi di join fare riferimento a http://en.wikipedia.org/wiki/Join_(SQL)
Usa questo:
SELECT p.name,(
SELECT COUNT(*) FROM [page-attachment] pa WHERE pa.[PAGE-ID] = p.id) as attachmentsnumber
FROM page p