最近工作中写了一触发器报错:ORA-04091:table XX  is mutating, trigger/function may not see it。

下面通过官方文档及网友提供资料分析一下错误原因及解决方法:

1.查看oracle官方文档:

原因:触发器(或者被语句中引用的用户自定义PL/SQL函数)视图去查询(或修改)一个被另一语句修改而触发的表。

解决方法:重写触发器(或函数)避免读该表。

2.根据错误原因我们写如下触发器,重现错误:

使用scott方案,创建一下表、触发器:

  1. SQL> create table tr_table as select * from emp;
  2. 表已创建。
  3. SQL> edit
  4. 已写入 file afiedt.buf
  5. 1  create or replace trigger tr_test
  6. 2  after update on emp
  7. 3  for each row
  8. 4  begin
  9. 5     update tr_table t set t.sal = (select sal from emp where empno=t.empno and empno=:new.empno);
  10. 6* end;
  11. SQL> /
  12. 触发器已创建
  13. SQL> update emp set sal=3700 where empno=7788;
  14. update emp set sal=3700 where empno=7788
  15. *
  16. 第 1 行出现错误:
  17. <span style="color:#ff0000;">ORA-04091: 表 SCOTT.EMP 发生了变化, 触发器/函数不能读它
  18. </span>ORA-06512: 在 "SCOTT.TR_TEST", line 2
  19. ORA-04088: 触发器 'SCOTT.TR_TEST' 执行过程中出错

3.原因分析:

在Oracle中执行DML语句的时候是需要显示进行提交操作的。当我们进行插入的时候,会触发触发器执行对触发器作用表和扩展表的种种操作,但是这个时候触发器和插入语句是在同一个事务管理中的,因此在插入语句没有被提交的情况下,我们无法对触发器作用表进行其他额外的操作。如果执行其他额外的操作则会抛出如上异常信息。

4.解决方案:

1) 我们知道,出错的原因是因为触发器和DML语句在同一事务管理中,所以方案一便是将触发器和DML语句分成两个单独的事务处理。这里可以使用Pragma autonomous_transaction; 告诉Oracle触发器是自定义事务处理。

SQL语句如下:

  1. create or replace trigger tr_test
  2. after update on emp
  3. for each row
  4. declare
  5. pragma autonomous_transaction;
  6. begin
  7. update tr_table t set t.sal = (select sal from emp where empno=t.empno and empno=:new.empno);
  8. commit; --此处需要显示提交
  9. end;

注:以上语句并不能实时获得更新的值。。。原因是我们在update emp表后还没来得及提交sal就触发了触发器,这个时候获取到的只能是老的sal值。

2) 在Oracle Trigger中有:new,:old两个特殊变量,当触发器为行级触发器的时候,触发器就会提供new和old两个保存临时行数据的特殊变量,我们可以从俩个特殊的变量中取出数据执行扩张表的DML操作。

SQL语句如下:

  1. create or replace trigger tr_test
  2. after update on emp
  3. for each row
  4. begin
  5. update tr_table t set t.sal = :new.sal;
  6. end;
  7. /

5. 再次插入数据:

  1. SQL> update emp set sal=3800 where empno=7788;
  2. 已更新 1 行。
  3. SQL> commit;
  4. 提交完成。
  1. SQL> select * from tr_table;
  2. EMPNO ENAME      JOB              MGR HIREDATE              SAL       COMM     DEPTNO
  3. ---------- ---------- --------- ---------- -------------- ---------- ---------- ----------
  4. 7369 SMITH      CLERK           7902 17-12月-80            800                    20
  5. 7499 ALLEN      SALESMAN        7698 20-2月 -81           1800        300         30
  6. 7521 WARD       SALESMAN        7698 22-2月 -81           1250        500         30
  7. 7566 JONES      MANAGER         7839 02-4月 -81           2975                    20
  8. 7654 MARTIN     SALESMAN        7698 28-9月 -81           1250       1400         30
  9. 7698 BLAKE      MANAGER         7839 01-5月 -81           2850                    30
  10. 7782 CLARK      MANAGER         7839 09-6月 -81           2450                    10
  11. <span style="BACKGROUND-COLOR: #009900">7788 SCOTT      ANALYST         7566 19-4月 -87           3800                    20
  12. </span>      7839 KING       PRESIDENT            17-11月-81           5000                    10
  13. 7844 TURNER     SALESMAN        7698 08-9月 -81           1500          0         30
  14. 7876 ADAMS      CLERK           7788 23-5月 -87           1100                    20
  15. EMPNO ENAME      JOB              MGR HIREDATE              SAL       COMM     DEPTNO
  16. ---------- ---------- --------- ---------- -------------- ---------- ---------- ----------
  17. 7900 JAMES      CLERK           7698 03-12月-81            950                    30
  18. 7902 FORD       ANALYST         7566 03-12月-81           3000                    20
  19. 7934 MILLER     CLERK           7782 23-1月 -82           1300                    10
  20. 已选择14行。

ORA-04091错误原因与解决方法的更多相关文章

  1. coreseek常见错误原因及解决方法

    coreseek常见错误原因及解决方法 Coreseek 中文全文检索引擎 Coreseek 是一款中文全文检索/搜索软件,以GPLv2许可协议开源发布,基于Sphinx研发并独立发布,专攻中文搜索和 ...

  2. /usr/bin/ld: cannot find -lc错误原因及解决方法

    问题解决 我在执行如下命令的时候,出现了错误. gcc -o main main.c -static -L. –lmylib Linux环境下gcc静态编译/usr/bin/ld: cannot fi ...

  3. SpringBoot+Shiro引起事务失效、错误原因、解决方法

    一.问题今天发现用户注册的Service的事务并没有起到作用,再抛出一个RuntimeException后,并没有发生回滚,下面是调试步骤: 1.检查数据库的引擎是否是innoDB 2.启动类上是否加 ...

  4. httpclient org.apache.http.NoHttpResponseException: host:端口 failed to respond 错误原因和解决方法

    原因:httpclient 之前与服务端建立的链接已经失效(例如:tomcat 默认的keep-alive timeout :20s),再次从连接池拿该失效链接进行请求时,就会保存. 解决方法:官方链 ...

  5. SyntaxError: (unicode error) 'unicodeescape' codec can't decode bytes in position 2-3: truncated \UXXXXXXXX escape错误原因及解决方法

    用Python打开文件: with open('C:\Users\PINPIN\test\file1.txt','r') as f2: pass 运行后直接就报错了: File "<i ...

  6. iBatis查询时报"列名无效"或"找不到栏位名称"无列名的错误原因及解决方法

    iBatis会自动缓存每条查询语句的列名映射,对于动态查询字段或分页查询等queryForPage, queryForList,就可能产生"列名无效".rs.getObject(o ...

  7. c++ std::sort函数调用经常出现的invalidate operator<错误原因以及解决方法

    在c++编程中使用sort函数,自定义一个数据结构并进行排序时新手经常会碰到这种错误. 这是为什么呢?原因在于什么?如何解决? 看下面一个例子: int main(int, char*[]) { st ...

  8. 移动端页头推荐配置 出现找不到favicon.ico错误原因和解决方法

    favicon 在未指定 favicon 时,大多数浏览器会请求 Web Server 根目录下的 favicon.ico .为了保证 favicon 可访问,避免404,必须遵循以下两种方法之一: ...

  9. javascript中出现identifier starts immediately after numeric literal错误原因以及解决方法

    javascript遇到参数是字符型和数字型组合的参数就会出现这种错误,例如alert(1);可以正確輸出alert(str);就會報錯alert("str");可以正確輸出.

随机推荐

  1. Flume 读取RabbitMq消息队列消息,并将消息写入kafka

    首先是关于flume的基础介绍 组件名称 功能介绍 Agent代理 使用JVM 运行Flume.每台机器运行一个agent,但是可以在一个agent中包含多个sources和sinks. Client ...

  2. python的模块与包的导入

    类似于C语言的包含头文件去引用其他文件的函数,python也有类似的机制,常用的引入方法有以下 import 模块名 #模块名就是py文件名 #使用这种方法以后调用函数的时候要使用模块名.函数名()这 ...

  3. LintCode题解之斐波纳契数列

    直接使用递归的方法会导致TLE,加个缓存就好了: public class Solution { private Integer[] buff = new Integer[1000]; /* * @p ...

  4. MongoDB 关系

    MongoDB 的关系表示多个文档之间在逻辑上的相互联系. 文档间可以通过嵌入和引用来建立联系. MongoDB 中的关系可以是: 1:1 (1对1) 1: N (1对多) N: 1 (多对1) N: ...

  5. dokcer自动化构建部署java web 基于jenkins+maven+nuxus容器

    # dokcer自动化构建部署java web 基于jenkins+maven+nuxus容器 #环境centos 7.4 docker 18.03.0-ce # nuxus,创建maven本地源(可 ...

  6. Docker部署Zabbix+Grafana监控

    Docker部署Zabbix+Grafana监控 环境 centos 7 ; Docker 17.12.0-ce ; docker-compose version 1.20.1 2018-4-1 当前 ...

  7. 数组中的数分为两组,让给出一个算法,使得两个组的和的差的绝对值最小,数组中的数的取值范围是0<x<100,元素个数也是大于0, 小于100 。

    比如a[]={2,4,5,6,7},得出的两组数{2,4,6}和{5,7},abs(sum(a1)-sum(a2))=0: 比如{2,5,6,10},abs(sum(2,10)-sum(5,6))=1 ...

  8. 将meteor部署到自己的服务器(deploy meteor to your own server)

    安装指定版本的node # 所有版本在:https://nodejs.org/download/release/# current dir:/rootwget -c https://nodejs.or ...

  9. Eric5 for Python 3.3.3安装指南

    一言蔽之,搭配是关键.以32位Window为例,先后安装: 1.PyQt PyQt4-4.10.3-gpl-Py3.3-Qt4.8.5-x32.exe http://www.riverbankcomp ...

  10. Hadoop学习笔记1:伪分布式环境搭建

    在搭建Hadoop环境之前,请先阅读如下博文,把搭建Hadoop环境之前的准备工作做好,博文如下: 1.CentOS 6.7下安装JDK , 地址: http://blog.csdn.net/yule ...