原文:在论坛中出现的比较难的sql问题:20(触发器专题2)


最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决的方法了。

所以,觉得有必要记录下来,这样以后再次碰到这类问题,也能从中获取解答的思路。

本篇是触发器专题,有很多触发器的问题。

1、关于触发器中回滚的问题。

在官方文档http://technet.microsoft.com/zh-cn/library/ms181299.aspx中“互操作性”段落中有这么一句:

    触发器继续执行 ROLLBACK 语句之后的所有其余语句。 如果这些语句中的任意语句修改数据,则不回滚这些修改。 执行其余的语句不会激发嵌套触发器。

其中“执行其余的语句不会激发嵌套触发器”,是否意味着在回滚操作后面的语句如果继续执行的话,后面语句触发的触发器就不会执行了。比如后面的语句有针对一个表的update操作,而这张表又有update的触发器,这个时候update会执行成功,但触发器不会执行?

但是很奇怪,我做了一个实验,发现,即使是嵌套触发器,也能够被触发,和微软文档上说的不一样。

从下面的insert时的输出,可以看出,在触发器中rollback之后的语句,是可以执行的,另外,update另一个表的操作也执行成功,并且触发了触发器。


  1. --drop table t1
  2. --drop table t2
  3. create table t1(id int)
  4. create table t2(id int)
  5. insert into t2
  6. values(100)
  7. go
  8. create trigger dbo.trigger_t1
  9. on t1
  10. for insert
  11. as
  12. rollback;
  13. select '这是rollback之后的语句,这里能执行'
  14. update t2
  15. set id = 1;
  16. go
  17. create trigger dbo.trigger_t2
  18. on t2
  19. for update
  20. as
  21. select '这是t2的update触发器,这里能执行'
  22. go
  23. --插入数据
  24. insert into t1
  25. values(1)
  26. /*
  27. 这是rollback之后的语句,这里能执行
  28. 这是t2的update触发器,这里能执行
  29. 消息 3609,级别 16,状态 1,第 3 行
  30. 事务在触发器中结束。批处理已中止。
  31. */
  32. --没有记录
  33. select * from t1
  34. select * from t2
  35. /*
  36. id
  37. 1
  38. */

2、触发器的错误处理

http://bbs.csdn.net/topics/390637035

我在A表写了个触发器 插入后执行,是向B表插入一条记录


  1. USE [test]
  2. GO
  3. SET ANSI_NULLS ON
  4. GO
  5. SET QUOTED_IDENTIFIER ON
  6. GO
  7. ALTER TRIGGER [dbo].[tba_delete]
  8. ON [dbo].[tba]
  9. AFTER INSERT
  10. AS
  11. BEGIN
  12. SET NOCOUNT ON;
  13. insert into test.dbo.tbb(title,info) values('title','info');
  14. END

在A表插入数据后正常。当我把 语言改为:

insert into test.dbo.tbb(id,title,info) values(1,'title','info');

故意出错。问题出现了,当触发器内的语句出错,向A表插入的数据就不能成功插入。

向朋友们求教个问题。我要如何把触发器中的出错信息记录下来?或者在什么地方可以查看到触发器执行时是否出错了,及查看错误的信息?

我的解法:


  1. USE [test]
  2. GO
  3. SET ANSI_NULLS ON
  4. GO
  5. SET QUOTED_IDENTIFIER ON
  6. GO
  7. create table [tba](title varchar(20),info varchar(20))
  8. create table tbb(title varchar(20),info varchar(20))
  9. --创建一个存放错误信息的表
  10. create table tb_error_message
  11. (
  12. obj_name nvarchar(30),
  13. obj_type nvarchar(15),
  14. err_msg nvarchar(100),
  15. err_date datetime
  16. )
  17. go
  18. drop trigger tba_delete
  19. go
  20. create TRIGGER [dbo].[tba_delete]
  21. ON [dbo].[tba]
  22. AFTER INSERT
  23. AS
  24. BEGIN
  25. SET NOCOUNT ON;
  26. begin try
  27. --这里模拟插入title列的数据长度,超过了定义时的20
  28. insert into test.dbo.tbb(title,info) values('title1111111111111111111','info');
  29. end try
  30. begin catch
  31. --错误了就回滚
  32. rollback;
  33. insert into tb_error_message
  34. values('tba_delete','trigger',error_message(),getdate())
  35. end catch
  36. END
  37. go
  38. insert into tba
  39. values('11','11')
  40. /*
  41. 消息 3930,级别 16,状态 1,过程 tba_delete,第 16 行
  42. 当前事务无法提交,而且无法支持写入日志文件的操作。请回滚该事务。
  43. 语句已终止。
  44. */
  45. --查询出错的信息
  46. select *
  47. from tb_error_message
  48. /*
  49. obj_name obj_type err_msg err_date
  50. tba_delete trigger 将截断字符串或二进制数据。 2013-11-08 12:32:16.750
  51. */

3、忽略触发器中的错误

http://bbs.csdn.net/topics/390623172

我的情况是这样的: 当向A表插入数据时,需要向某用户发送信息通知,由于插入数据方是第三方软件做的,我控件不了,所以我在A表加个触发器来实现,现在问题是发送信息这里的错误是可以忽略的,就是说假如发送失败不影响数据的插入,请问我在触发器里如何忽略错误,让事务继续执行.

我的思路是:


  1. --先保存一个事务点
  2. SAVEPOINT xxx
  3. begin try
  4. --或者是调用存储过程,或者就直接写代码
  5. exec 发送信息的存储过程
  6. end try
  7. begin catch
  8. select '执行失败'
  9. --一旦出错,只是回滚到上面的事务点,继续执行
  10. ROLLBACK [WORK] TO SAVEPOINT xxx
  11. end catch
  12. --其他代码

发布了416 篇原创文章 · 获赞 135 · 访问量 94万+

在论坛中出现的比较难的sql问题:20(触发器专题2)的更多相关文章

  1. 在论坛中出现的比较难的sql问题:46(日期条件出现的奇怪问题)

    原文:在论坛中出现的比较难的sql问题:46(日期条件出现的奇怪问题) 最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决的方法了. 所以,觉得有 ...

  2. 在论坛中出现的比较难的sql问题:45(用户在线登陆时间的小时、分钟计算问题)

    原文:在论坛中出现的比较难的sql问题:45(用户在线登陆时间的小时.分钟计算问题) 最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决的方法了. ...

  3. 在论坛中出现的比较难的sql问题:44(触发器专题 明细表插入数据时调用主表对应的数据)

    原文:在论坛中出现的比较难的sql问题:44(触发器专题 明细表插入数据时调用主表对应的数据) 最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决 ...

  4. 在论坛中出现的比较难的sql问题:42(动态行转列 考勤时间动态列)

    原文:在论坛中出现的比较难的sql问题:42(动态行转列 考勤时间动态列) 所以,觉得有必要记录下来,这样以后再次碰到这类问题,也能从中获取解答的思路.

  5. 在论坛中出现的比较难的sql问题:41(循环替换 循环替换关键字)

    原文:在论坛中出现的比较难的sql问题:41(循环替换 循环替换关键字) 所以,觉得有必要记录下来,这样以后再次碰到这类问题,也能从中获取解答的思路.

  6. 在论坛中出现的比较难的sql问题:40(子查询 销售和历史库存)

    原文:在论坛中出现的比较难的sql问题:40(子查询 销售和历史库存) 最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决的方法了. 所以,觉得有 ...

  7. 在论坛中出现的比较难的sql问题:39(动态行转列 动态日期列问题)

    原文:在论坛中出现的比较难的sql问题:39(动态行转列 动态日期列问题) 最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决的方法了. 所以,觉 ...

  8. 在论坛中出现的比较难的sql问题:38(字符拆分 字符串检索问题)

    原文:在论坛中出现的比较难的sql问题:38(字符拆分 字符串检索问题) 最近,在论坛中,遇到了不少比较难的sql问题,虽然自己都能解决,但发现过几天后,就记不起来了,也忘记解决的方法了. 所以,觉得 ...

  9. 在论坛中出现的比较难的sql问题:37(动态行转列 某一行数据转为列名)

    原文:在论坛中出现的比较难的sql问题:37(动态行转列 某一行数据转为列名) 所以,觉得有必要记录下来,这样以后再次碰到这类问题,也能从中获取解答的思路.

  10. 在论坛中出现的比较难的sql问题:36(动态行转列 解析json格式字符串)

    原文:在论坛中出现的比较难的sql问题:36(动态行转列 解析json格式字符串) 所以,觉得有必要记录下来,这样以后再次碰到这类问题,也能从中获取解答的思路.

随机推荐

  1. linux: 右键添加打开终端

    安装一个包,即可在右键里面添加一个“打开终端”的菜单. sudo apt-get install nautilus-open-terminal 注销用户重启,然后再进入就可以右键->在终端打开选 ...

  2. Oracle 查询表分区相关信息

    Oracle 查询表分区相关信息 --表分区 --1,分区表信息 -- (1)显示数据库所有分区表的信息 select * from DBA_PART_TABLES a where a.owner=u ...

  3. rapidjson的简单使用(转)

    rapidjson的简单使用 C++ rapidjson 基础入门 rapidjson图文讲解

  4. 利用OpenGL固定流水线绘制球体

    在OS X上的一个OpenGL简单demo.所附赠的代码是绘制半个球体.开启了深度缓存和多重采样,采样数是4. 详细下载地址请见:http://www.cocoachina.com/bbs/read. ...

  5. window.location.href重定向失败的问题

    如题,在js中通过window.location.href=URL来跳转到另一个页面(也可以是另一个项目的另一个页面). 打开的页面地址是:www.a.com/project1/index 要跳转的页 ...

  6. python多线程中join()的理解

    在 Python 的多线程编程中,经常碰到 thread.join()这样的代码.那么今天咱们用实际代码来解释一下 join 函数的作用. 第一,当一个进程启动之后,会默认产生一个主线程,因为线程是程 ...

  7. 【438】Python 处理文件

    1. 读取文件,计算 tweets 数目 python中readline判断文件读取结束的方法 line == '' python:如何检查一行是否为空行 line == '\n' or line = ...

  8. 报错:java.lang.ClassNotFoundException: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException

    1.创建mysql_link的时候一定要注意是不是多加了空格: 2.检查Sqoop是否引入了jdbc的jar包.

  9. Ubuntu18.04 instsall XMind_8 and crack

    1.dowload XMind_8 linux install zip wget https://www.xmind.cn/xmind/downloads/xmind-8-update8-linux. ...

  10. 一种计算hash的思路

    /*** * 转换请求hash,根据转换模式计算hash,防止重复发送请求,浪费服务器资源(内存.cpu.文件系统等) * @param mode * @param data * @return */ ...