oracle - ORA-04091: 表 [blah] 正在发生变化,触发器/函数可能看不到它

标签 oracle hibernate triggers ora-04091

我最近开始处理一个大型复杂应用程序,由于此错误,我刚刚被分配了一个错误:

ORA-04091: table SCMA.TBL1 is mutating, trigger/function may not see it
ORA-06512: at "SCMA.TRG_T1_TBL1_COL1", line 4
ORA-04088: error during execution of trigger 'SCMA.TRG_T1_TBL1_COL1'

有问题的触发器看起来像
    create or replace TRIGGER TRG_T1_TBL1_COL1
   BEFORE  INSERT OR UPDATE OF t1_appnt_evnt_id ON TBL1
   FOR EACH ROW
   WHEN (NEW.t1_prnt_t1_pk is not  null)
   DECLARE
        v_reassign_count number(20);
   BEGIN
       select count(t1_pk) INTO v_reassign_count from TBL1
              where  t1_appnt_evnt_id=:new.t1_appnt_evnt_id and t1_prnt_t1_pk is not null;
       IF (v_reassign_count > 0) THEN
           RAISE_APPLICATION_ERROR(-20013, 'Multiple reassignments not allowed');
       END IF;
   END;

该表有一个主键“t1_pk”,一个“约会事件ID”t1_appnt_evnt_id和另一列“t1_prnt_t1_pk”可能或可能
不包含另一行的 t1_pk .

似乎触发器正试图确保没有其他人拥有
t1_appnt_evnt_id已经引用了同一行,如果这一行引用另一行,则该行引用了对另一行的引用。

DBA 对错误报告的评论说“删除触发器,并在代码中执行检查”,但不幸的是,他们在 Hibernate 之上有一个专有的代码生成框架,所以我什至无法弄清楚它实际上在哪里被写出来,所以我希望有一种方法可以使这个触发器起作用。在那儿?

最佳答案

我想我不同意你对触发器试图做什么的描述
做。在我看来,它旨在强制执行此业务规则:对于
给定 t1_appnt_event 的值,只有一行可以具有非 NULL 值
t1_prnt_t1_pk 一次。 (它们在第二列中是否具有相同的值并不重要。)

有趣的是,它是为 UPDATE OF t1_appnt_event 定义的,而不是为其他列定义的,所以我认为有人可以通过更新第二列来打破规则,除非该列有单独的触发器。

可能有一种方法可以创建一个强制执行此规则的基于函数的索引,以便您可以完全摆脱触发器。我想出了一种方法,但它需要一些假设:

  • 该表有一个数字主键
  • 主键和 t1_prnt_t1_pk 都是正数

  • 如果这些假设成立,您可以创建这样的函数:
    dev> create or replace function f( a number, b number ) return number deterministic as
      2  begin
      3    if a is null then return 0-b; else return a; end if;
      4  end;
    

    和这样的索引:
    CREATE UNIQUE INDEX my_index ON my_table
      ( t1_appnt_event, f( t1_prnt_t1_pk, primary_key_column) );
    

    因此 PMNT 列为 NULL 的行将出现在索引中,主键的倒数作为第二个值,因此它们永远不会相互冲突。非 NULL 的行将使用列的实际(正)值。如果两行在两列中具有相同的非 NULL 值,那么您可能会违反约束的唯一方法。

    这可能过于“聪明”,但它可能会帮助您解决问题。

    来自 Paul Tomblin 的更新:我对 igor 在评论中提出的原始想法进行了更新:
     CREATE UNIQUE INDEX cappec_ccip_uniq_idx 
     ON tbl1 (t1_appnt_event, 
        CASE WHEN t1_prnt_t1_pk IS NOT NULL THEN 1 ELSE t1_pk END);
    

    关于oracle - ORA-04091: 表 [blah] 正在发生变化,触发器/函数可能看不到它,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/375968/

    相关文章:

    Oracle 日期舍入

    sql - QUERY 查找表中日期之间的差距

    .net - 将 .NET DateTime 转换为 OracleDate

    java - 最佳数据库迁移方法?

    java - JPA 延迟加载适用于最终实体和私有(private)构造函数

    Mysql从varchar列复制到decimal列

    sql - 当我一次又一次用相同的值更新表时会发生什么?

    c# - .Net 中的 Oracle 连接池

    java - Spring Hibernate 查询中的日期

    php - 从 MySQL 触发器调用 PHP 脚本