使用游标多次更新 pgsql

Posted

技术标签:

【中文标题】使用游标多次更新 pgsql【英文标题】:multiple update pgsql using cursor 【发布时间】:2017-02-02 06:11:30 【问题描述】:

此代码没有错误,但没有任何数据更改,请帮助我。我想从一些库存表中多次更新数量

drop FUNCTION SP_PROSES_STOCK(noresep bigint, p_post_cd varchar);
CREATE or replace FUNCTION SP_PROSES_STOCK(noresep bigint, p_post_cd varchar)
RETURNS void
LANGUAGE plpgsql
as $function$
DECLARE cursorData refcursor;
    v_item_cd varchar;
    v_quantity numeric;

begin

open cursorData FOR
select A.item_cd, A.quantity from trx_medical_resep B
inner join trx_resep_data A on A.medical_resep_seqno = B.medical_resep_seqno
where B.medical_resep_seqno = noresep; 
fetch next from cursorData into v_item_cd,v_quantity;
while (found)
loop
    update inv_pos_item set quantity = quantity - v_quantity 
    where item_cd = v_item_cd and pos_cd = p_post_cd;
end loop;
close cursorData;

END 
$function$

【问题讨论】:

【参考方案1】:

我修改了您的函数以使其更短并使用 FOR。我还添加了 RAISE NOTICE 进行调试。可以试试吗:

CREATE or replace FUNCTION SP_PROSES_STOCK(noresep bigint, p_post_cd varchar)
RETURNS void
LANGUAGE plpgsql
as $function$
DECLARE v_cursor record;

BEGIN

   FOR v_cursor IN
       SELECT A.item_cd,
              A.quantity
         FROM trx_medical_resep B
         JOIN trx_resep_data A ON A.medical_resep_seqno = B.medical_resep_seqno
        WHERE B.medical_resep_seqno = noresep
   LOOP
       RAISE NOTICE 'Changes for %', v_curosr; 
       UPDATE inv_pos_item
           SET quantity = quantity - v_cursor.quantity 
         WHERE item_cd = v_cursor.item_cd
           AND pos_cd = p_post_cd;
   END LOOP;
END 
$function$

调试完成后,您可以删除 RAISE NOTICE。

【讨论】:

【参考方案2】:

你不需要循环。单个 UPDATE 语句会快很多:

CREATE or replace FUNCTION SP_PROSES_STOCK(noresep bigint, p_post_cd varchar)
  RETURNS void
as 
$function$
begin

  update inv_pos_item 
     set quantity = quantity - v.quantity 
  from (
    select A.item_cd, A.quantity 
    from trx_medical_resep B
      join trx_resep_data A on A.medical_resep_seqno = B.medical_resep_seqno
    where B.medical_resep_seqno = noresep
  ) v
  where item_cd = v.item_cd and pos_cd = p_post_cd;

END;
$function$
LANGUAGE plpgsql;

【讨论】:

谢谢大家,我的数据库有数据错误,还有字符串\r的添加,不知道怎么会这样,我用dbeaver输入数据样本【参考方案3】:

有些场景需要光标。例如,假设您正在更新 1000 万行,并且在第 6 百万条记录处引发异常,更新将失败,然后它将回滚在失败之前更新的所有行。如果使用游标,它会变慢,但您可以更好地控制该过程,并能够绕过错误并继续发布更新。但话说回来,你的里程可能会有所不同......

【讨论】:

以上是关于使用游标多次更新 pgsql的主要内容,如果未能解决你的问题,请参考以下文章

数据库错题库第八章8.4(游标可以多次使用,多次打开和关闭!)

使用新游标多次执行 CKQueryOperation

pgsql游标插入数据失败

关于 Pl/pgsql 中游标的一些错误

用于行 mod 3 的 PL/pgSQL 游标

mysql游标的用法及作用