SQL-INSERT触发器练习
&练习一
有这样的一个基础表A,字段包括:id、type、value、create_time,主要是记录某个类型的状态变化时间和值。在插入类型(type)为‘runtime’ 的数据时,根据前后变化的值,更新一条该type的前后两条时间组合的信息到另一张表B。
定义:value为布尔型,true-->false时,为停止时间,false-->true时,为运行时间
具体要求:表A插入一条runstate的数据时,需要在表B插入一条停止或运行的信息。
eg:
A表
B表
触发器如下(如有问题,欢迎指出):
1 CREATE TRIGGER RUN_STATE
2 ON A
3 FOR INSERT
4 AS
5 IF( (SELECT COUNT(1) FROM inserted) > 1 )
6 BEGIN
7 RAISERROR('一次只能插入一条数据!',16,10);
8 rollback;
9 END
10 ELSE
11 BEGIN
12 IF( (SELECT TOP 1 type FROM inserted)='runtime' ) ----插入的数据是否为runtime状态信息
13 BEGIN
14 IF EXISTS ( SELECT 1 FROM B WHERE type = 'runtime') -----B表中是否已经创建了runtime信息表
15 BEGIN
16 DECLARE @Old_state varchar(10),@create_time smalldatetime;
17 SET @Old_state = (SELECT state FROM B WHERE type = 'runtime');
18 SET @Create_time = (SELECT TOP 1 create_time FROM inserted);
19 if @Old_state = '运行'
20 BEGIN
21 update B SET end_time = @Create_time WHERE type = 'runtime';
22 update B SET state = @Create_time WHERE type = '停止';
23 END
24 ELSE
25 BEGIN
26 update B SET start_time = @Create_time WHERE type = 'runtime';
27 update B SET state = @Create_time WHERE type = '运行';
28 END
29 END
30 else
31 BEGIN
32 DECLARE @id int,@type varchar(20),@value bit;
33 SELECT TOP 1 @id=id,@type=type,@value=value,@create_time=create_time FROM inserted;
34 IF @value = 1
35 INSERT B (id,type,start_time,state) values(@id,'runtime',@create_time,'运行');
36 ELSE
37 INSERT B (id,type,end_time,state) values(@id,'runtime',@create_time,'停止');
38 END
39 END
40 END
41 GO
&练习二
需要用到的表如下(卷烟库存表【CI_list】,卷烟销售表【CS_list】):
--业务规则:库存金额 = 库存数量 * 库存单价
--业务规则:销售金额 = 销售数量 * 销售单价
eg1:强制执行业务规则,保证插入的数据中,库存金额 = 库存数量 * 库存单价,且库存金额不为空
触发器如下(如有问题,欢迎指出):
1 CREATE TRIGGER T_INSERT_CS
2 ON CS_list
3 FOR INSERT
4 AS
5 DECLARE @i_num int,@i_uprice money,@i_amount money;
6 IF ( (SELECT COUNT(*) FROM inserted)>1 )
7 BEGIN
8 RAISERROR('一次只能插入一行数据',16,10);
9 ROLLBACK;
10 END
11 ELSE
12 BEGIN
13 SELECT TOP 1 @i_num=i_num,@i_uprice=i_uprice,@i_amount=i_amount FROM inserted;
14 IF ISNULL(@i_amount,1)=1 OR @i_amount != @i_num*@i_uprice
15 BEGIN
16 RAISERROR('插入的数据不符合业务规则!已根据规则修改数据',16,10);
17 UPDATE CS_list SET i_amount = @i_num*@i_uprice WHERE cs_brand IN (SELECT TOP 1 cs_brand FROM inserted)
18 END
19 END
20 GO
21
22 INSERT INTO CS_list values('红塔山新势力',2,30,60);
23 INSERT INTO CS_list (cs_brand,i_num,i_uprice) values('红1',2,20);
24 INSERT INTO CS_list values('云南映像',2,30,70);
25 GO
eg2:如果销售的卷烟品牌不存在库存或者库存为零,则返回错误;否则则自动减少[卷烟库存表]中对应品牌卷烟的库存数量和库存金额。并且满足业务规则
触发器如下(如有问题,欢迎指出):
1 CREATE TRIGGER T_INSERT_CI
2 ON CI_list
3 FOR INSERT
4 AS
5 IF ( (SELECT COUNT(*) FROM inserted)>1 )
6 BEGIN
7 RAISERROR('一次只能插入一行数据',16,10);
8 ROLLBACK;
9 END
10 ELSE
11 BEGIN
12 DECLARE @CI_BRAND varchar(40),@CS_BRAND varchar(40);
13 DECLARE @s_num int,@s_uprice money,@s_amount money;
14 SELECT TOP 1 @CI_BRAND=ci_brand,@s_num=s_num,@s_uprice=s_uprice,@s_amount=s_amount FROM inserted;
15 IF not EXISTS (SELECT 1 FROM CS_list WHERE cs_brand=@CI_BRAND)
16 BEGIN17 RAISERROR('库存中不含当前要销售的卷烟',16,10);
18 ROLLBACK;
19 END
20 ELSE IF (SELECT i_num FROM CS_list WHERE cs_brand=@CI_BRAND) = 0
21 BEGIN
22 RAISERROR('库存中已经没有当前要销售的香烟',16,10);
23 ROLLBACK;
24 END
25 ELSE
26 BEGIN
27 IF ISNULL(@s_amount,1)=1 OR @s_amount != @s_num*@s_uprice
28 BEGIN
29 RAISERROR('插入的数据不符合业务规则!已根据规则修改数据',16,10);
30 UPDATE CI_list SET s_amount = @s_num*@s_uprice WHERE ci_brand = (SELECT TOP 1 ci_brand FROM inserted)
31 END
32 UPDATE CS_list SET i_num=i_num-@s_num WHERE cs_brand = (SELECT TOP 1 ci_brand FROM inserted);
33 UPDATE CS_list SET i_amount=i_num*i_uprice WHERE cs_brand = (SELECT TOP 1 ci_brand FROM inserted);
34 END
35 END
&练习三
如下所示三张表( student,grade,student_updata_before ):
student表
grade表
Student_update_before表
触发器需要实现一下要求:
如果学生所在的班级存在,则班级表学生数+1。
如果学生所在的班级不存在,则在班级表中新建班级,并将该学生加入到该班级中。
触发器如下(如有问题,欢迎指出):
1 CREATE TRIGGER Add_stu
2 ON student
3 FOR INSERT
4 AS
5 IF ( SELECT COUNT(1) FROM inserted ) > 1
6 BEGIN
7 RAISERROR('每次只能插入一条数据',16,10);
8 ROLLBACK;
9 END
10 ELSE
11 BEGIN
12 DECLARE @Sg_name varchar(30);
13 SELECT TOP 1 @Sg_name = Sg_name FROM inserted;
14 IF EXISTS ( SELECT 1 FROM grade WHERE Gname = @Sg_name )
15 BEGIN
16 DECLARE @num1 int;
17 SET @num1 = (SELECT COUNT(*) FROM student WHERE Sg_name = @Sg_name);
18 UPDATE grade SET Gnum = @num1 WHERE Gname = @Sg_name;
19 END
20 ELSE
21 BEGIN
22 DECLARE @gid int;
23 IF (SELECT COUNT(1) FROM grade) = 0
24 SET @gid = 1;
25 ELSE
26 SET @gid = (SELECT MAX(Gid) FROM grade ) + 1;
27 declare @num int;
28 INSERT INTO grade VALUES(@gid,@Sg_name,1,'');
29 END
30 END
31 GO
&练习四
#准备工作:创建员工employee表、部门变动历史dept_history表,工资变动历史sal_history表
#1,创建员工employee表:
1 CREATE TABLE employee(
2 eid varchar(20) PRIMARY KEY,
3 ename varchar(20),
4 dept varchar(20),
5 salary int,
6 uptime date,
7 status int
8 )
9 go
#2,创建部门变动历史dept_history表
CREATE TABLE dept_history(
did int identity(1,1) primary key,
eid varchar(20),
olddept varchar(20),
newdept varchar(20),
uptime date,
FOREIGN KEY(eid) REFERENCES employee(eid)
)
go
#3,创建工资变动历史sal_history表
1 CREATE TABLE sal_history(
2 sid int identity(1,1) primary key,
3 eid varchar(20),
4 oldsal varchar(20),
5 newsal varchar(20),
6 uptime date,
7 FOREIGN KEY(eid) REFERENCES employee(eid)
8 )
9 go
触发器实现功能:当新职工入职时,员工信息表将插入1条数据。同时,触发器在部门变动历史中增加1条记录,其中olddept值为null;在工资变动历史中增加1条记录,其中oldsal值为0。
触发器如下(如有问题,欢迎指出):
CREATE TRIGGER Add_em
ON employee
FOR INSERT
AS
IF ( SELECT COUNT(1) FROM inserted ) > 1
BEGIN
RAISERROR('每次只能插入一条数据!请重新输入!',16,10)
END
ELSE
BEGIN
DECLARE @eid varchar(20),@dept varchar(20),@salary int;
SELECT TOP 1 @eid = eid ,@dept = dept ,@salary = salary FROM inserted;
INSERT INTO dept_history (eid,olddept,newdept,uptime) values(@eid,null,@dept,GETDATE());
INSERT INTO sal_history (eid,oldsal,newsal,uptime) values(@eid,0,@salary,GETDATE());
END
SQL-INSERT触发器练习的更多相关文章
- 记一次SQL Server Insert触发器编写过程
实现功能:新增特定类型的新闻时,自动追加特定的背景图片. 第一版(错误信息:不能在 'inserted' 表和 'deleted' 表中使用 text.ntext 或 image 列),代码如下: - ...
- SQL server触发器中 update insert delete 分别给写个例子被。
SQL server触发器中 update insert delete 分别给写个例子以及解释下例子的作用和意思被, 万分感谢!!!! 主要想知道下各个语句的书写规范. INSERT: 表1 (ID, ...
- 在Sql Server触发器中判断操作是Insert还是Update还是Delete
在Sql Server触发器中判断操作是Insert还是Update还是Delete DECLARE @IsInsert bit, @IsUpdate bit, @IsDelete ...
- SQL Server触发器
一﹕ 触发器是一种特殊的存储过程﹐它不能被显式地调用﹐而是在往表中插入记录﹑更新记录或者删除记录时被自动地激活.所以触发器可以用来实现对表实施复杂的完整性约`束. 二﹕ SQL Server为每个触发 ...
- SQL Server 触发器
触发器是一种特殊类型的存储过程,它不同于之前的我们介绍的存储过程.触发器主要是通过事件进行触发被自动调用执行的.而存储过程可以通过存储过程的名称被调用. Ø 什么是触发器 触发器对表进行插入.更新.删 ...
- [SQL] SQL Server 触发器
触发器是一种特殊类型的存储过程,它不同于之前的我们介绍的存储过程.触发器主要是通过事件进行触发被自动调用执行的.而存储过程可以通过存储过程的名称被调用. Ø 什么是触发器 触发器对表进行插入.更新.删 ...
- SQL Server 触发器(转)
触发器是一种特殊类型的存储过程,它不同于之前的我们介绍的存储过程.触发器主要是通过事件进行触发被自动调用执行的.而存储过程可以通过存储过程的名称被调用. Ø 什么是触发器 触发器对表进行插入.更新.删 ...
- SQL Server 触发器【转】
触发器是一种特殊类型的存储过程,它不同于之前的我们介绍的存储过程.触发器主要是通过事件进行触发被自动调用执行的.而存储过程可以通过存储过程的名称被调用. Ø 什么是触发器 触发器对表进行插入.更新.删 ...
- Oracle数据库编程:使用PL/SQL编写触发器
8.使用PL/SQL编写触发器: 触发器存放在数据缓冲区中. 触发器加序列能够实现自动增长. 在触发器中不能使用connit和rollback. DML触发器 ...
- sql server触发器的例子
发布:thebaby 来源:脚本学堂 [大 中 小] 本文介绍下,在sql server数据库中使用触发器的简单例子,有需要的朋友可以参考下,希望对你有一定的帮助. 原文地址:http:/ ...
随机推荐
- SQL语句(三)分组函数和分组查询
目录 一.分组函数 特点 1. 各函数的简单使用 2. 搭配distinct的使用 3. COUNT 统计行数 4. 和分组函数一同查询的字段要求是group by后的字段 二.分组查询 1. 简单应 ...
- Git 修改历史 commits 中的用户名和邮箱
一.作用 修改某个仓库历史 commit 的用户 name 和 email 信息. 将历史提交记录中的指定 name/email 修改为新的 name/email. 二.步骤 确认本地全局邮箱/用户名 ...
- css文件编码
当css文件的编码
- CTF_论剑场_Web20
直接上脚本,多跑几次就能出flag import requests import re url = "http://123.206.31.85:10020/" s = reques ...
- linux50个常用命令
1.存放用户账号的文件在哪里? /etc/passwd 2.如何删除一个非空的目录? rm -rf 目录名 3.查看当前的工作目录用什么命令? pwd 4.创建一个文件夹用什么命令? mkdir 5. ...
- Nacos 笔记
Nacos 笔记 目录 Nacos 笔记 1. Nacos简介 1.1 主流配置中心对比 1.2 主流注册中心对比 1.3 Nacos特性 2. 安装启动 支持外部 MySQL 3. 配置管理 3.1 ...
- 用AutoHotkey做汉字到Unicode字符串的转换
要把汉字转换为搜的形式,也就是在汉字的Unicode Big Endian编码前面加"&#x",后面加分号.例如""字转换后为"搜" ...
- 用华为云cli,管理华为云服务器的,安全组端口
---[前言]--- 关键字 hcloud 华为 命令行 linux windows powershell 前些天,大家因为华为云,是否应该默认开启端口,大家吵起来了,所以我抽空写了此文.解决问题,缓 ...
- noip模拟12[简单的区间·简单的玄学·简单的填数]
noip模拟12 solutions 这次考试靠的还是比较好的,但是还是有不好的地方, 为啥嘞??因为我觉得我排列组合好像白学了诶,文化课都忘记了 正难则反!!!!!!!! 害没关系啦,一共拿到了\( ...
- 三个线程按循序一个打印A一个打印B一个打印C 循环打印?
第一种 public static volatile int flag = 1; public static void printABC1(){ Thread t1 = new Thread(() - ...