在存储过程内的Postgres 8.4中转义LIKE模式或regexp字符串

时间:2015-04-02 11:30:15

标签: regex postgresql stored-procedures plpgsql

我正在编写一个存储过程来查找主项的子项并进行更新。场景就像这样

Id  Item Name               Parent Id  
1   Item A 14.1             NULL  
2   Item B 14.1.1           1      
3   Item C 14.1.2           1       
4   Item B 14.1.3           1       
5   Item A 14.1.1.1         2 

我在SO String matching in PostgreSQL 8.4上发布了另一个问题,以获取某个项目的子项。根据该答案,我必须逃避项目代码,并且必须在查询中使用它。但我还没有办法逃脱这种情况。 SP如下:

CREATE OR REPLACE FUNCTION updateitemparentnew(bigint, int) RETURNS text
    LANGUAGE plpgsql STRICT
    AS $$
DECLARE
    itemdetail RECORD;    
    codeSearch text;
    codeEscapedSearch text;
    result text;
BEGIN   

--Get th details of the current item
SELECT INTO itemdetail * FROM om_item WHERE item_id = $1;

codeSearch = itemdetail.item_code||'.';
codeEscapedSearch = codeSearch;  --Need to be corrected. It should escape the item code

-- Event 1=> add 2 => edit 3 => delete
IF $2 = 1 THEN
    --Find new children and update then
    result =  'UPDATE om_item SET item_parentid = '||itemdetail.item_id
           ||' WHERE item_id IN (
                 SELECT item_id FROM om_item
                 WHERE  item_code LIKE \''||codesearch||'%\'
                 AND item_code ~ \''||codeEscapedsearch||'[^.]+$\');';

END IF;

return result;
END;
$$;

在查询中codeEscapedSearch应该进行转义,以处理代码本身中的.和正则表达式中的.

2 个答案:

答案 0 :(得分:1)

考虑这种直接方法:

CREATE OR REPLACE FUNCTION updateitemparentnew(_id bigint, _operation text)
  RETURNS void LANGUAGE plpgsql STRICT AS
$func$
DECLARE
   code_like text;
   code_regex text;
BEGIN

SELECT INTO code_like, code_regex
       p.item_code || '.%'
     , '^' || replace(p.item_code, '.', '\.') || '\.[^.]+$'
FROM   om_item p
WHERE  p.item_id = _id;

CASE _operation  -- ins / upd / del
WHEN 'upd' THEN  -- Find new children and update then
   UPDATE om_item c
   SET    item_parentid = _id
   WHERE  c.item_code LIKE code_like
   AND    c.item_code ~ code_regex;

-- WHEN 'ins' THEN ...
-- WHEN 'del' THEN ...
END CASE;

END
$func$;

这不会返回查询字符串,而是直接执行UPDATE。更短更快。

还使用replace(),顺便说一句。

答案 1 :(得分:0)

由于没有找到适当的解决方案,我决定使用简单的replace
codeEscapedSearch = replace(codeSearch,'.','\\.');