Pull to refresh

Comments 8

CREATE OR REPLACE FUNCTION replace_multiple(input_string TEXT, replacements TEXT[])
  RETURNS TEXT AS
$$
DECLARE
  i INT;
BEGIN
  IF cardinality(replacements) % 2 <> 0 THEN
    RAISE EXCEPTION 'Number of replacements must be even';
  END IF;
  
  FOR i IN 1..cardinality(replacements)/2 LOOP
    input_string := replace(input_string, replacements[(i-1)*2+1], replacements[i*2]);
  END LOOP;
  
  RETURN input_string;
END;
$$
LANGUAGE plpgsql;

Да, но та же проблема с повторной заменой.

Надо уточнять начальные условия.


  1. Строка 'ab', подстановки {'а' -> 'x', 'b' -> 'y', 'ab' -> 'z'}. Какой результат ожидается?
  2. Строка 'ab', подстановка {'' -> 'x'}. Какой результат ожидается?
  3. Какие символы допустимы в изначальной строке? Какие символы допустимы в подстановках?

А если еще вспомнить, что '.' в regexp совсем не то же самое, что в качестве подстроки...

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

Можно значительно проще и экономнее на regexp

Пример на MySQL (на PG думаю будет аналогично):

WITH d AS (
    SELECT 'abcdaaabbbcccdcba' s
    , 'a,x;bb,y;ccc,z;' repls
    , 'xbcdxxxybzdcbx' test_res
)
, d2 AS (
    SELECT *, CONCAT(s, '\n', repls) s_ex
    FROM d
)
, d3 AS (
    SELECT *
    , SUBSTRING_INDEX(REGEXP_REPLACE(s_ex, '(?<search>[^\n]+)(?=.*?(\n|\n.*?;)\\\k<search>,(?<repl>[^;]+);)', '${repl}', 1, 0, 'm'), '\n', 1) res
    FROM d2 d
)
SELECT s, res
, res = test_res is_correct
, VERSION() ver
FROM d3

+-----------------+--------------+----------+-----+
|s                |res           |is_correct|ver  |
+-----------------+--------------+----------+-----+
|abcdaaabbbcccdcba|xbcdxxxybzdcbx|1         |8.3.0|
+-----------------+--------------+----------+-----+

Способ любопытный. Правда, опирается на отсутствие \n в исходной строке и ; в заменах.

не проблема сгенерить 3 уникальных разделителя
- между строкой и всеми заменами
- между поиском и заменой
- между группами замен
и использовать их для построения регулярки

Нашёл подобное приемлемое решение на MySQL как закапиталайзить предложение

WITH d AS (
    SELECT 'Lorem Ipsum is simply dummy text of the printing and typesetting industry.' s
)
, d2 AS (
    SELECT *
    , CONCAT(LOWER(s), UPPER(s)) s_l_u
    , CONCAT('(?<=[[:space:]]|^)[a-z](?=.{', (CHAR_LENGTH(s) - 1), '}([A-Z]))') reg_exp
    , CHAR_LENGTH(s) len
    FROM d
)
SELECT version() ver, s, SUBSTR(REGEXP_REPLACE(s_l_u, reg_exp, '$1'), 1, len) cap
FROM d2

+------+--------------------------------------------------------------------------+--------------------------------------------------------------------------+
|ver   |s                                                                         |cap                                                                       |
+------+--------------------------------------------------------------------------+--------------------------------------------------------------------------+
|8.0.35|Lorem Ipsum is simply dummy text of the printing and typesetting industry.|Lorem Ipsum Is Simply Dummy Text Of The Printing And Typesetting Industry.|
+------+--------------------------------------------------------------------------+--------------------------------------------------------------------------+

Sign up to leave a comment.