Как применить функцию к каждому элементу столбца массива в Postgres? - PullRequest
45 голосов
/ 21 декабря 2011

Pg-запрос возвращает массив. Я хотел бы получить это с каждым элементом, отформатированным до 3 десятичных знаков. Как я могу применить функцию к каждому элементу массива? Что-то вроде следующего (неправильно, очевидно) -

SELECT Round(ARRAY[1.53224,0.23411234], 2);
{1.532, 0.234}

Наверное, я ищу что-то вроде map функции Perl.

Ответы [ 4 ]

94 голосов
/ 21 декабря 2011

Сначала превратите массив в набор, используя unnest :

> SELECT n FROM unnest(ARRAY[1.53224,0.23411234]) AS n;
     n      
------------
    1.53224
 0.23411234
(2 rows)

Затем примените выражение к столбцу:

> SELECT ROUND(n, 2) FROM unnest(ARRAY[1.53224,0.23411234]) AS n;
 round 
-------
  1.53
  0.23
(2 rows)

Наконец, используйте array_agg , чтобы превратить набор обратно в массив:

> SELECT array_agg(ROUND(n, 2)) FROM unnest(ARRAY[1.53224,0.23411234]) AS n;
  array_agg  
-------------
 {1.53,0.23}
(1 row)
28 голосов
/ 21 декабря 2011
postgres=# select array(select round(unnest(array[1.2,2.4,3,4])));
   array   
-----------
{1,2,3,4}
(1 row)
15 голосов
/ 21 декабря 2011

Возможно, вам потребуется создать сохраненную функцию.Вот тот, который делает то, что вам нужно:

CREATE OR REPLACE FUNCTION array_round(float[], int)
RETURNS float[]
AS
$$
DECLARE
   arrFloats ALIAS FOR $1;
   roundParam ALIAS FOR $2;
   retVal float[];
BEGIN
   FOR I IN array_lower(arrFloats, 1)..array_upper(arrFloats, 1) LOOP
    retVal[I] := round(CAST(arrFloats[I] as numeric), roundParam);
   END LOOP;
RETURN retVal;
END;
$$
LANGUAGE plpgsql 
   STABLE 
RETURNS NULL ON NULL INPUT;

Затем вызовите что-то вроде этого:

# SELECT array_round(ARRAY[1.53224,0.23411234], 2);
 array_round 
-------------
 {1.53,0.23}
(1 row)
0 голосов
/ 21 декабря 2011

Вам нужно превратить массив в набор строк.Например, используя generate_series:

SELECT ARRAY(SELECT ROUND(ARRAY[1.53224,0.23411234])[i], 2) FROM generate_series(1,2) AS s(i));    

Я знаю, это довольно уродливо.Должна быть вспомогательная функция, чтобы сделать такие сопоставления проще.

Возможно, что-то вроде (да, это ужасный, медленный и хрупкий динамический код):

CREATE OR REPLACE FUNCTION map_with_arg(TEXT, ANYARRAY, TEXT)
RETURNS ANYARRAY
IMMUTABLE STRICT
LANGUAGE 'plpgsql' AS
$$
DECLARE
    i INTEGER;
    t TEXT;
    cmd TEXT;
BEGIN
    FOR i IN array_lower($2, 1) .. array_upper($2, 1) LOOP
        cmd := 'SELECT ('||quote_ident($1)||'('||quote_nullable($2[i])||', '||quote_nullable($3)||'))::TEXT';
        EXECUTE cmd INTO t;
        $2[i] := t;
    END LOOP;
    RETURN $2;
END;
$$;

select map_with_arg('repeat', array['can','to']::TEXT[], '2');
 map_with_arg
---------------
 {cancan,toto}

Обновление Мне кажется, что мы могли бы использовать один динамический оператордля всего цикла.Это может смягчить некоторые проблемы с производительностью.

CREATE OR REPLACE FUNCTION map_with_arg(TEXT, ANYARRAY, TEXT)
RETURNS ANYARRAY
IMMUTABLE STRICT
LANGUAGE 'plpgsql' AS
$$
DECLARE
    cmd TEXT;
    rv TEXT;
BEGIN
    cmd := 'SELECT ARRAY(SELECT (' || quote_ident($1)||'($1[i], '||quote_nullable($3)||'))::TEXT FROM generate_subscripts($1, 1) AS gs(i))';
    EXECUTE cmd USING $2 INTO rv;
    RETURN rv;
END;
$$;
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...