PostgreSQL
 sql >> Teknologi Basis Data >  >> RDS >> PostgreSQL

Mengkueri jumlah baris yang dipisahkan berdasarkan rentang tanggal

crosstab() fungsi dengan dua parameter.

Harus bekerja seperti ini, untuk mendapatkan nilai untuk 2012:

SELECT * FROM crosstab(
     $$SELECT testname, to_char(last_update, 'mon_YYYY'), count(*)::int AS ct
        FROM   tests
        WHERE  current_status = 'FAILED'
        AND    last_update >= '2012-01-01 0:0'
        AND    last_update <  '2013-01-01 0:0'  -- proper date range!
        GROUP  BY 1,2
        ORDER  BY 1,2$$

    ,$$VALUES
      ('jan_2012'::text), ('feb_2012'), ('mar_2012')
    , ('apr_2012'), ('may_2012'), ('jun_2012')
    , ('jul_2012'), ('aug_2012'), ('sep_2012')
    , ('oct_2012'), ('nov_2012'), ('dec_2012')$$)
AS ct (testname  text
   , jan_2012 int, feb_2012 int, mar_2012 int
   , apr_2012 int, may_2012 int, jun_2012 int
   , jul_2012 int, aug_2012 int, sep_2012 int
   , oct_2012 int, nov_2012 int, dec_2012 int);

Temukan penjelasan mendetail di bawah pertanyaan terkait ini.

Saya tidak menguji. Seperti yang dikomentari @Craig, nilai sampel akan membantu.
Diuji sekarang dengan kasus pengujian saya sendiri.

Jangan tampilkan nilai NULL

Masalah utama (bahwa bulan tanpa baris tidak akan muncul sama sekali) dihindari oleh crosstab() fungsi dengan dua parameter.

Anda tidak dapat menggunakan COALESCE dalam kueri dalam, karena NULL nilai dimasukkan oleh crosstab() diri. Anda bisa ...

1. Bungkus semuanya menjadi subquery:

SELECT testname
      ,COALESCE(jan_2012, 0) AS jan_2012
      ,COALESCE(feb_2012, 0) AS feb_2012
      ,COALESCE(mar_2012, 0) AS mar_2012
      , ...
FROM (
    -- query from above)
    ) x;

2. LEFT JOIN kueri utama ke daftar lengkap bulan.

Dalam hal ini, Anda tidak memerlukan parameter kedua menurut definisi.
Untuk rentang yang lebih besar, Anda dapat menggunakan generate_series() untuk membuat nilai.

SELECT * FROM crosstab(
     $$SELECT t.testname, m.mon, count(x.testname)::int AS ct
       FROM  (
          VALUES
           ('jan_2012'::text), ('feb_2012'), ('mar_2012')
          ,('apr_2012'), ('may_2012'), ('jun_2012')
          ,('jul_2012'), ('aug_2012'), ('sep_2012')
          ,('oct_2012'), ('nov_2012'), ('dec_2012')
       ) m(mon)
       CROSS JOIN (SELECT DISTINCT testname FROM tests) t
       LEFT JOIN (
          SELECT testname
                ,to_char(last_update, 'mon_YYYY') AS mon
          FROM   tests
          WHERE  current_status = 'FAILED'
          AND    last_update >= '2012-01-01 0:0'
          AND    last_update <  '2013-01-01 0:0'  -- proper date range!
          ) x USING (mon)
       GROUP  BY 1,2
       ORDER  BY 1,2$$
     )
AS ct (testname  text
   , jan_2012 int, feb_2012 int, mar_2012 int
   , apr_2012 int, may_2012 int, jun_2012 int
   , jul_2012 int, aug_2012 int, sep_2012 int
   , oct_2012 int, nov_2012 int, dec_2012 int);

Kasus uji dengan data sampel

Berikut ini adalah kasus uji dengan beberapa data sampel yang gagal disediakan oleh OP. Saya menggunakan ini untuk mengujinya dan membuatnya bekerja.

CREATE TEMP TABLE tests (
  id             bigserial PRIMARY KEY
 ,testname       text NOT NULL
 ,last_update    timestamp without time zone NOT NULL DEFAULT now()
 ,current_status text NOT NULL
 );

INSERT INTO tests (testname, last_update, current_status)
VALUES
  ('foo', '2012-12-05 21:01', 'FAILED')
 ,('foo', '2012-12-05 21:01', 'FAILED')
 ,('foo', '2012-11-05 21:01', 'FAILED')
 ,('bar', '2012-02-05 21:01', 'FAILED')
 ,('bar', '2012-02-05 21:01', 'FAILED')
 ,('bar', '2012-03-05 21:01', 'FAILED')
 ,('bar', '2012-04-05 21:01', 'FAILED')
 ,('bar', '2012-05-05 21:01', 'FAILED');



  1. Database
  2.   
  3. Mysql
  4.   
  5. Oracle
  6.   
  7. Sqlserver
  8.   
  9. PostgreSQL
  10.   
  11. Access
  12.   
  13. SQLite
  14.   
  15. MariaDB
  1. Bagaimana saya bisa memulai server PostgreSQL di Mac OS X?

  2. DROP TABLE JIKA ADA Contoh di PostgreSQL

  3. Pencarian teks Django dengan pembaruan pencocokan kalimat parsial ke django3

  4. Bagaimana cara menerapkan fungsi ke setiap elemen kolom array di Postgres?

  5. Bagaimana cara menyapu db:drop dan menyapu db:create di Heroku?