Вы можете определить свою собственную функцию агрегирования, а затем использовать ее со спецификацией окна, чтобы получить агрегированный вывод на каждом этапе, а не одно значение.
Таким образом, агрегат представляет собой часть состояния и функцию преобразования для изменения этого состояния для каждой строки, а также, при необходимости, функцию финализации для преобразования состояния в выходное значение. Для такого простого случая, как этот, достаточно просто использовать функцию преобразования.
create function ema_func(numeric, numeric) returns numeric
language plpgsql as $$
declare
alpha numeric := 0.5;
begin
-- uncomment the following line to see what the parameters mean
-- raise info 'ema_func: % %', $1, $2;
return case
when $1 is null then $2
else alpha * $2 + (1 - alpha) * $1
end;
end
$$;
create aggregate ema(basetype = numeric, sfunc = ema_func, stype = numeric);
, что дает мне:
steve@steve@[local] =# select x, ema(x, 0.1) over(w), ema(x, 0.2) over(w) from data window w as (order by n asc) limit 5;
x | ema | ema
-----------+---------------+---------------
44.988564 | 44.988564 | 44.988564
39.5634 | 44.4460476 | 43.9035312
38.605724 | 43.86201524 | 42.84396976
38.209646 | 43.296778316 | 41.917105008
44.541264 | 43.4212268844 | 42.4419368064
Кажется, что эти цифры соответствуют таблице, которую вы добавили к вопросу.
Кроме того, вы можете определить функцию для передачи альфы в качестве параметра из оператора:
create or replace function ema_func(state numeric, inval numeric, alpha numeric)
returns numeric
language plpgsql as $$
begin
return case
when state is null then inval
else alpha * inval + (1-alpha) * state
end;
end
$$;
create aggregate ema(numeric, numeric) (sfunc = ema_func, stype = numeric);
select x, ema(x, 0.5 /* alpha */) over (order by n asc) from data
Кроме того, эта функция на самом деле настолько проста, что вообще не должна быть в plpgsql, но может быть просто функцией sql, хотя вы не можете ссылаться на параметры по имени в одном из них:
create or replace function ema_func(state numeric, inval numeric, alpha numeric)
returns numeric
language sql as $$
select case
when $1 is null then $2
else $3 * $2 + (1-$3) * $1
end
$$;