Rumah > pangkalan data > tutorial mysql > Bagaimana untuk Mengira Baris Sebelumnya Dengan Cekap Dalam Julat Masa dalam PostgreSQL?

Bagaimana untuk Mengira Baris Sebelumnya Dengan Cekap Dalam Julat Masa dalam PostgreSQL?

Linda Hamilton
Lepaskan: 2024-12-27 07:27:12
asal
454 orang telah melayarinya

How to Efficiently Count Preceding Rows Within a Time Range in PostgreSQL?

Mengira Baris Sebelumnya dalam Julat

Pernyataan Masalah:

Tentukan jumlah kiraan rekod sebelumnya dalam masa yang ditetapkan julat untuk setiap baris dalam jadual.

Khusus Senario:

Pertanyaan:

SELECT id, date
     , count(*) OVER (HAVING previous_rows.date >= (date - '1 hour'::interval))  -- ?
FROM test;
Salin selepas log masuk

Jadual:

CREATE TABLE test (
  id  bigint
, ts  timestamp
);
Salin selepas log masuk

Postgres 11 atau Lebih Baru:

Postgres 11 memperkenalkan pilihan pembingkaian fungsi tetingkap yang dipertingkatkan, membenarkan penggunaan mod RANGE dengan MEndahului dan MENGIKUTI untuk memilih baris dalam ofset yang ditentukan.

SELECT id, ts
     , count(*) OVER (ORDER BY ts RANGE '1 hour' PRECEDING EXCLUDE CURRENT ROW)
FROM   test
ORDER  BY ts;
Salin selepas log masuk

Postgres 10 atau Lebih:

ROM (pertanyaan Rom):

SELECT id, ts
     , (SELECT count(*)::int - 1
        FROM   unnest(dates) x
        WHERE  x >= sub.ts - interval '1h') AS ct
FROM (
   SELECT id, ts
        , array_agg(ts) OVER(ORDER BY ts) AS dates
   FROM   test
   ) sub;
Salin selepas log masuk

ARR (tatasusunan kiraan elemen):

SELECT id, ts
     , (SELECT count(*)
        FROM   test t1
        WHERE  t1.ts >= t.ts - interval '1h'
        AND    t1.ts < t.ts) AS ct
FROM   test t
ORDER  BY ts;
Salin selepas log masuk

COR (subkueri berkorelasi):

CREATE OR REPLACE FUNCTION running_window_ct(_intv interval = '1 hour')
  RETURNS TABLE (id bigint, ts timestamp, ct int)
  LANGUAGE plpgsql AS
$func$
DECLARE
   cur   CURSOR FOR
         SELECT t.ts + _intv AS ts1
              , row_number() OVER (ORDER BY t.ts ROWS UNBOUNDED PRECEDING) AS rn
         FROM   test t
         ORDER  BY t.ts;
   rec   record;
   rn    int;
BEGIN
   OPEN cur;
   FETCH cur INTO rec;
   ct := -1;  -- init

   FOR id, ts, rn IN
      SELECT t.id, t.ts
           , row_number() OVER (ORDER BY t.ts ROWS UNBOUNDED PRECEDING)
      FROM   test t ORDER BY t.ts
   LOOP
      IF rec.ts1 >= ts THEN
         ct := ct + 1;
      ELSE
         LOOP
            FETCH cur INTO rec;
            EXIT WHEN rec.ts1 >= ts;
         END LOOP;
         ct := rn - rec.rn;
      END IF;

      RETURN NEXT;
   END LOOP;
END
$func$;
Salin selepas log masuk

Panggil fungsi:

SELECT * FROM running_window_ct();
Salin selepas log masuk

Keputusan Penanda Aras:

Tanda aras menggunakan jadual dengan kiraan baris yang berbeza-beza menunjukkan bahawa fungsi FNC adalah pemenang yang jelas dari segi prestasi dan kebolehskalaan.

Atas ialah kandungan terperinci Bagaimana untuk Mengira Baris Sebelumnya Dengan Cekap Dalam Julat Masa dalam PostgreSQL?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:php.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan