文章概要:

某项目将数据从 SQLSERVER 迁移到 KES。其中SQLSERVER中触发器用到了 TRIGGER_NESTLEVEL() 函数,KES并不能直接支持该函数。

起初在分析该问题时想复杂了本文做了一次记录。实际上在kes兼容sqlsevrer基础语法,直接简单使用SYS_TRIGGER_DEPTH()替换 TRIGGER_NESTLEVEL() 函数即可,用来判断递归层级实测是等价的(如果只是想知道最终解决方案,读到这里就可以了)。

本文将主要介绍该函数结合客户代码的替代改造分析过程和解决方案验证。

一,TRIGGER_NESTLEVEL() 函数的作用是撒?

1,查官网的解释:返回为激发触发器的语句执行的次数。 TRIGGER_NESTLEVEL 在 DML 和 DDL 触发器中用以确定当前的嵌套层数。

官网链接:https://learn.microsoft.com/zh-cn/sql/t-sql/functions/trigger-nestlevel-transact-sql?view=sql-server-ver16

2,同时也翻阅了一下百度,对于DDL触发器而言,一般都是用于防止触发器一直触发,例如触发器里面执行数据DELETE,

再触发当前触发器,从而导致进入循环,所以要如果当前触发内会对表再执行DELETE,

需要添加对TRIGGER_NESTLEVEL的检查,防止触发器嵌套层数太多。

3,并且对于触发器本身而言,需要参数RECURSIVE_TRIGGERS开启后触发器才能递归(默认情况下不递归,这也是在测试验证过程发现的)

官网链接:https://learn.microsoft.com/zh-cn/sql/relational-databases/triggers/create-nested-triggers?view=sql-server-ver16

这是客户代码的sqlserver代码的简化案例:

create trigger "tgr_delete_name"
on "dbo".tbl_xxx_name
for delete
as
declare @id int
begin
select @id =id from deleted
if ( (select trigger_nestlevel() ) <30 and @@rowcount >0)
begin
delete from budg_code_item where super_id = @id
end
end;

触发器本身而言逻辑很简单,是一个delete事件触发器,作用于tbl_xxx_name表,在满足递归级别为30以内时,且存在数据被删除(@@rowcount >0)的条件时,又在触发器体内对tbl_xxx_name表执行delete,从而形成递归,

trigger_nestlevel的作用就是限制递归层数,如果没有这条IF语句则会死循环下去。

二,TRIGGER_NESTLEVEL() 函数的改造分析

那么现在的问题是,KES支持这么玩吗?KES 中的递归检测又是什么?

KES支持触发器递归:

在PL/SQL中,触发器可以形成递归。递归触发器是指一个触发器在执行期间触发了另一个相同类型的触发器。这种情况可能会导致无限循环和性能问题。

为了避免或者限制递归触发器,查阅一些资料,可以使用以下方法:

1)禁用触发器:在触发器中添加条件,只有当满足某些条件时才执行触发器的操作。这样可以避免触发器在执行期间再次触发自己。

2)使用标志变量:在触发器中使用一个标志变量来标记触发器是否已经执行过。如果触发器已经执行过,则不再执行触发器的操作。

3)调整触发器顺序:如果有多个触发器与同一表相关联,可以调整它们的执行顺序,确保递归触发器不会发生。

4)重新设计触发器:如果递归触发器无法避免,可能需要重新设计数据库模型或修改触发器逻辑,以避免递归触发器的情况。

需要注意的是,递归触发器可能会导致性能问题和无限循环,因此在设计和使用触发器时需要小心处理,确保其行为可控和可预测。

一开始思考改造问题时,觉得第二条应该可以(或者1和2结合),但是一细想当存在各种情况的并发时,该方法就无法满足了。

3和4也不能满足,因为无法得到递归层级,因此上述方法都不可取。

但是,但是,实际上上面的考虑想复杂了,KES支持sys_trigger_depth()函数来判断触发器的递归

三,TRIGGER_NESTLEVEL() 函数的改造结果及其验证

create table tt1(id int , supid int , na varchar(10));	

insert into tt1 values
( 1 , 1 , 'a'),
(11 , 1 , 'a11'),
(12 , 11 , 'a12'),
(111 ,12, 'b111'),
(112 ,111, 'b12'),
(121 ,122, 'c21'),
(122 ,121, 'c22'); create trigger "tgr_delete_name" ---kes的POC分支上支持该语法,可以实现sqlserevr基本语法的兼容
on tt1
for delete
as
declare @id int
begin
select @id = id from deleted
if ( (select sys_trigger_depth() ) < 4 and @@rowcount > 0 )
begin
delete from tt1 where supid = @id
end
end; delete from tt1 where id = 1; select * from tt1; test-# delete from tt1 where id = 1;
test-# /
DELETE 1
test-# select * from tt1;
test-# /
id | supid | na
----+-------+-----
112 | 111 | b12
121 | 122 | c21
122 | 121 | c22
(3 rows)

sqlserver运行验证:

USE master;
GO
ALTER DATABASE master SET RECURSIVE_TRIGGERS ON;
GO
drop table tt1;
go
create table tt1(id int , supid int , na varchar(10));
go
insert into tt1 values
( 1 , 1 , 'a'),
(11 , 1 , 'a11'),
(12 , 11 , 'a12'),
(111 ,12, 'b111'),
(112 ,111, 'b12'),
(121 ,122, 'c21'),
(122 ,121, 'c22');
go create trigger tgr_delete_name
on tt1
for delete
as
declare @id int
begin
select @id =id from deleted
if ( (select trigger_nestlevel() ) < 4 and @@rowcount >0)
begin
delete from tt1 where supid = @id
end
end;
go
delete from tt1 where id = 1; select * from tt1; --运行结果 112 111 b12
121 122 c21
122 121 c22

其余验证不再累述,结果一致,改造完毕。

改造小结

实际上,查微软的官网来看TRIGGER_NESTLEVEL()函数的功能要强大一些,但是仅就判触发器递归层级而言,sys_trigger_depth()可以直接替换 TRIGGER_NESTLEVEL() 函数。

另一种思考,基于客户递归删除数据的逻辑,是否可以使用如下WITH RECURSIVE AS语句递归删除?该语句也是能控制递归层数的,不过暂未尝试过。

WITH RECURSIVE CTE_NAME AS (
初始语句(非递归部分)
UNION ALL
递归部分语句
)
[ SELECT| INSERT | UPDATE | DELETE]

SQLServer递归触发器在KES中的一次改造分析的更多相关文章

  1. SQLServer之触发器简介

    触发器定义 触发器是数据库服务器中发生事件时自动执行的一种特殊存储过程.SQLServer允许为任何特定语句创建多个触发器.它的执行不是由程序调用,也不是手工启动,而是由事件来触发,当对数据库进行操作 ...

  2. sqlserver数据库触发器调用外部exe

    sqlserver数据库触发器调用外部exe,同事可以选择参入参数! sqlserver使用 master..xp_cmdshell 进行外部exe的执行. 使用master..xp_cmdshell ...

  3. 在SQLServer使用触发器实现数据完整性

    1.实现数据完整性的手段 在sqlserver中,在服务器端实现数据完整性主要有两种手段:一种是在创建表时定义数据完整性,主要分为:实体完整性.域完整性.和级联参照完整性:实现的手段是创建主键约束.唯 ...

  4. 分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间)

    分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间) 很多时候我们都需要计算数据库中各个表的数据量和每行记录所占用空间 这里共享一个脚本 CREATE TABLE #tab ...

  5. (转)分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间)

    分享一个SQLSERVER脚本(计算数据库中各个表的数据量和每行记录所占用空间) 很多时候我们都需要计算数据库中各个表的数据量和每行记录所占用空间 这里共享一个脚本 CREATE TABLE #tab ...

  6. SqlServer添加触发器死锁的原因

    之前遇到过SqlServer添加触发器死锁的情况,纠结了很长时间 最近发现原来是因为我在建表的时候,把id设成主键后,系统默认了加一个聚集的索引 就是聚集索引把表锁住了

  7. java基础 File 递归删除文件夹中所有文件文件夹 目录(包含子目录)下的.java文件复制到e:/abc文件夹中, 并统计java文件的个数

    File 递归删除文件夹中所有文件文件夹 package com.swift.kuozhan; import java.io.File; import java.util.Scanner; /*键盘录 ...

  8. day18 时间:time:,日历:calendar,可以运算的时间:datatime,系统:sys, 操作系统:os,系统路径操作:os.path,跨文件夹移动文件,递归删除的思路,递归遍历打印目标路径中所有的txt文件,项目开发周期

    复习 ''' 1.跨文件夹导包 - 不用考虑包的情况下直接导入文件夹(包)下的具体模块 2.__name__: py自执行 '__main__' | py被导入执行 '模块名' 3.包:一系列模块的集 ...

  9. C中二叉排序树的非递归和递归插入操作以及中序遍历代码实现【可运行】

    C中二叉排序树的非递归和递归插入操作以及中序遍历代码实现[可运行] #include <stdio.h> #include <stdlib.h> typedef int Key ...

  10. 城市经纬度 json 理解SignalR Main(string[] args)之args传递的几种方式 串口编程之端口 多线程详细介绍 递归一个List<T>,可自己根据需要改造为通用型。 Sql 优化解决方案

    城市经纬度 json https://www.cnblogs.com/innershare/p/10723968.html 理解SignalR ASP .NET SignalR 是一个ASP .NET ...

随机推荐

  1. rpm的一些命令

    rpm -q xx #查询当前的包是否安装 rpm -qi xx # 查询当前包的详细信息 rpm -qpi 包文件路径 # 没装之前先查看包的信息 rpm -qpl 包文件路径 # 预计装上后会在系 ...

  2. java基础之StringBuilder---03

    StringBuilder概述 StringBuilder是一个可变的字符串类,我们可以把它看成是一个容器,这里的可变指的是StringBuilder对象中的内容是可变的. 如果对字符串进行拼接操作, ...

  3. java+mysql学生信息管理系统

    实现:mysql+eclipse(idea设置之后也可运行)+jdk8 功能: 管理员:管理登+管理员注册 学生:添加学生信息+删除学生信息+修改学生信息+查询学生信息+学生列表展示 界面展示: 详情 ...

  4. 【LeetCode回溯算法#01】图解组合问题

    组合问题 力扣题目链接(opens new window) 给定两个整数 n 和 k,返回范围 [1, n] 中所有可能的 k 个数的组合. 示例: 输入: n = 4, k = 2 输出: [ [2 ...

  5. ZYNQ核心板及其底板开源啦!

    Hello-FPGA ZYNQ 设计开源啦! 开源ZYNQ核心板 + 底板 硬件设计.软件设计,软件设计使用裸机演示,演示了如何使用AXI DMA等关键dma 模块 欢迎加QQ 讨论 94755958 ...

  6. Java面向对象之内部类的几类使用场景

    介绍 Java内部类是一种特殊的类,它定义在另一个类的内部.内部类提供了许多有用的特性,包括访问外部类的私有成员.隐藏实现细节以及实现回调接口等.以下是Java内部类的一些常用场景及其举例说明: 回调 ...

  7. 【Azure 应用服务】Web.config中设置域名访问限制,IP地址限制访问特定的页面资源 (Rewrite)

    问题描述 问题一:web app已经绑定了域名,例如是www.a.com,现在只允许使用www.a.com 访问,如果使用默认的域名xxxx.chinacloundsites.cn访问的时候,需要显示 ...

  8. 【Azure 应用服务】如何让App Service 支持 Delete 方法 

    问题描述 如何让webapp 支持 delete 方法? 在不修改设置的情况下,调用DELETE方法出现405错误 - 方法不被允许 问题解决 基于当前App Service在Windows的环境中运 ...

  9. 【Azure 应用服务】App Service 使用Tomcat运行Java应用,如何设置前端网页缓存的相应参数呢(-Xms512m -Xmx1204m)?

    问题描述 App Service 使用Tomcat运行Java应用,如何设置前端网页缓存的相应参数呢(-Xms512m -Xmx1204m)? 问题回答 App Service在Windows的环境中 ...

  10. D3.js 力导向图的显示优化(二)- 自定义功能

    摘要: 在本文中,我们将借助 D3.js 的灵活性这一优势,去新增一些 D3.js 本身并不支持但我们想要的一些常见的功能:Nebula Graph 图探索的删除节点和缩放功能. 文章首发于 Nebu ...