PostgreSQL
 sql >> Database >  >> RDS >> PostgreSQL

Perché PostgreSQL combina le serie in modo sbagliato?

È una stranezza di come vengono eseguite più funzioni di restituzione di set quando vengono invocate in SELECT -elenco. Ti aspetti che il risultato sia il prodotto incrociato dei due, ma non è così che funziona. In realtà è il multiplo comune più basso del numero di righe dei due.

Vedi:

Confronta:

test=>     SELECT generate_series(1,3) aval, generate_series(1,4) bval;
 aval | bval 
------+------
    1 |    1
    2 |    2
    3 |    3
    1 |    4
    2 |    1
    3 |    2
    1 |    3
    2 |    4
    3 |    1
    1 |    2
    2 |    3
    3 |    4
(12 rows)

test=>     SELECT generate_series(1,3) aval, generate_series(1,3) bval;
 aval | bval 
------+------
    1 |    1
    2 |    2
    3 |    3
(3 rows)

Per questo motivo, su PostgreSQL 9.2 e versioni successive dovresti usare LATERAL query in cui invochi funzioni di restituzione di set nel FROM clausola:

test=>     SELECT aval, bval FROM generate_series(1,3) aval CROSS JOIN LATERAL generate_series(1,3) bval;
 aval | bval 
------+------
    1 |    1
    1 |    2
    1 |    3
    2 |    1
    2 |    2
    2 |    3
    3 |    1
    3 |    2
    3 |    3
(9 rows)

test=>     SELECT aval, bval FROM generate_series(1,3) aval CROSS JOIN LATERAL generate_series(1,4) bval;
 aval | bval 
------+------
    1 |    1
    1 |    2
    1 |    3
    1 |    4
    2 |    1
    2 |    2
    2 |    3
    2 |    4
    3 |    1
    3 |    2
    3 |    3
    3 |    4
(12 rows)

Nelle versioni precedenti puoi utilizzare una sottoquery-in-FROM per evitare di avere più SRF in un termine SELECT:

test=> SELECT generate_series(1,3) aval, bval FROM (SELECT generate_series(1,4)) AS x(bval);
 aval | bval 
------+------
    1 |    1
    2 |    1
    3 |    1
    1 |    2
    2 |    2
    3 |    2
    1 |    3
    2 |    3
    3 |    3
    1 |    4
    2 |    4
    3 |    4
(12 rows)