SQLserver 环境下触发器以及级联更新与删除
该文触发器是根据本作者对中学学籍管理系统项目数据库需求分析后得出,已经实验论证完毕,以下是实例的创建以及触发器的实现过程,本作者的百度文库中目录下可以找到数据库课程设计文档中可供参考,该文档涉及本系统的系统从头到尾的设计需求以及数据库实施等阶段一直到数据库的应用,以及数据库中的三级模式两级映像。
中学学籍管理系统数据库物理逻辑结构:
use master
--创建数据库
CREATE DATABASE SDbase
ON PRIMARY--主文件组
(
NAME='SDbase_Data',--逻辑文件名
FILENAME='D:\SQLSDbase\SDbase_Data.mdf',
SIZE=5MB,--初始大小
MAXSIZE=100MB,--最大容量
FILEGROWTH=15%--增长率
)
LOG ON--日志文件
(
NAME='SDbase_Log',--逻辑文件名
FILENAME='D:\SQLSDbase\SDbase_Log.ldf',
SIZE=5MB,
FILEGROWTH=0--未启用自动增长
)
GO
中学学籍管理系统表创建:
use SDbase
--1创建学校信息表
Create table SchoolInfor
(ScName VARCHAR(20) , --学校名称
Sprovince VARCHAR(20) not null,--学校所在省份
SbureauEd VARCHAR(20) not null,--学校所属教育局
Saddress varchar(50) not null,--学校地址
PRIMARY KEY(ScName)
)
--1部别信息表
Create table BubieTable(Bno char(1) unique ,Bname char(4) primary key)
--1年级信息表
Create table NianjiTable(Nno char(1) unique ,Nname char(6) primary key)
--1毕业生信息表
Create table BiyeSheng(
Sno char(12),
buBie char(10),
Sname varchar(36),
primary key(Sno)
)
--2班级表
Create table classTbale --班级表
(buBie char(4) , --类别初中、高中、小学等等
nianJi char(6),
banJi char(6) not null, --班级编号
TeacherNo varchar(12) unique NOT NULL, --老师编号
TeacherName varchar(50) not NULL,
primary key(buBie,nianJi,banJi),
foreign key(buBie) references BubieTable(Bname),
foreign key(nianJi) references NianjiTable(Nname),
)
--3入学信息表
Create table StudentInfor --创建学生入学基本信息表
(enrollmentYear char(4) not null,--考虑到年份都是四位数,所以定死为char(4)类型和大小
Sno char(12), --学生学号
buBie char(4) , --入学类型(初中、高中)
nianJi char(6),
banJi char(6),
Sname varchar(50) not null, --考虑到少数名族名字
Ssex char(2) check(Ssex in('男','女')),
Nationnality varchar(30) not null, --民族
placeBirth varchar(20)not null, --籍贯
birthDate Date not null, --出生日期
graduatedUniversity VARCHAR(20) not null,--毕业学校
enrollmentScore smallint check(enrollmentScore>=0),--总分
sourceUpdate char(10) not null, --来源
muqianZhuangtai Varchar(30), --变动原因
PRIMARY KEY (Sno),
foreign key(buBie) references BubieTable(Bname),
foreign key(nianJi) references NianjiTable(Nname),
--主键
)
--4创建学生个人情况
Create table StudentGerenInfor
(Sno char(12), --学号
mianMao char(12) not null, --政治面貌
shenTi char(10) , --身体状况
jiaTingRenKou char(2), --家庭人口
homeAdress VARCHAR(30) not null,--家庭住址
TemlePhone char(11) not null, --联系方式11位的手机号
PRIMARY KEY(Sno),
FOREIGN KEY (Sno) references StudentInfor(Sno)
)
--4创建学生主要关系表
Create table SGuanxi
(Sno char(12),
chengWei char(4), --与本人关系。,
Name varchar(50) NOT NULL, --姓名
Age CHAR(3) NOT NULL, --年龄
DanWei VARCHAR(50)NOT NULL, --单位
PRIMARY KEY (Sno,chengWei,Name), --考虑到一个人有多个家庭成员,但是可以有相同称谓的姐姐、弟弟,标识唯一一行
FOREIGN KEY (Sno) references StudentInfor(Sno)
)
--4学籍更改表
Create table xuejiUpdate
(
Sno char(12), --学号
updateType char(4), --变更类型
updateDate DATETime, --变更日期
yuanYin varchar(50), --变更原因
SendOrYanZheng VARCHAR(50),--发给或缴纳证件
PRIMARY KEY(Sno,updateDate),
foreign key (Sno) references StudentInfor(Sno)
)
--4创建系统用户
Create table UserType
(userID varchar(12), --用户名
userPassWord varchar(18), --账户实际操作人姓名
name varchar(50), --密码
dengJi char(1), --权限等级
shuoMing varchar(50), --权限说明
primary key (userID) --主键
)
数据的载入:
use SDbase
--插入学校信息
--插入学校信息(学校名称、所在省份、所属教育局、学校地址)
insert into SchoolInfor values('黄石理工附中','湖北省','黄石市教育局','黄石市下陆区团城山街道')
insert into SchoolInfor values('黄石二中','湖北省','黄石市教育局','黄石市下陆区扬州路')
-- select*from SchoolInfor
--插入学习类别
insert into BubieTable values('1','小学');
insert into BubieTable values('2','初中');
insert into BubieTable values('3','高中');
--插入年级
insert into NianjiTable values('1','一年级');
insert into NianjiTable values('2','二年级');
insert into NianjiTable values('3','三年级');
insert into NianjiTable values('4','四年级');
insert into NianjiTable values('5','五年级');
insert into NianjiTable values('6','六年级');
--查询年级
select * from NianjiTable
--为学校新增建制班,同时为该班级指定一名班主任,后经过触发器为该班主任提供一个权限账户来登录系统来管理本班学生
insert into classTbale values('小学','一年级','1','01','张三');
insert into classTbale values('小学','二年级','2','02','李四');
insert into classTbale values('小学','三年级','3','03','王五');
insert into classTbale values('初中','一年级','1','04','李大勇');
insert into classTbale values('初中','二年级','2','05','李晓勇');
insert into classTbale values('初中','三年级','3','06','李勇');
insert into classTbale values('高中','一年级','1','07','张十三');
insert into classTbale values('高中','二年级','2','08','张山');
insert into classTbale values('高中','三年级','3','09','王麻子');
--学校录取学生时候,由招生办老师提供的名单,当录取学生的时候,
--同时为该学生提供一个系统账户,账户名为学生的学号,密码默认为123456,账户类型为学生账户
insert into StudentInfor values('2020','202020491111','初中','一年级','1','余欣','女','汉族','襄阳','1999-01-28','湖北理工','480','录取','无');
insert into StudentInfor values('2016','201616491110','初中','二年级','2','张一','男','汉族','恩施','1997-11-06','湖北理工','490','录取','无');
insert into StudentInfor values('2017','201717491101','初中','三年级','3','张二','女','汉族','十堰','1998-01-01','湖北理工','510','休学','无');
insert into StudentInfor values('2020','202020491155','高中','一年级','1','张四','女','汉族','宣恩','2000-01-11','湖北理工','489','休学','无');
insert into StudentInfor values('2017','201717491170','高中','二年级','2','张五','女','汉族','洛阳','1997-12-25','湖北理工','490','录取','无');
insert into StudentInfor values('2018','201616491180','高中','三年级','3','张六','女','汉族','商丘','1998-03-03','湖北理工','470','录取','无');
--录入学生的个人情况
insert into StudentGerenInfor values('202020491111','团员','健康','5','襄阳','10010100861');
insert into StudentGerenInfor values('201616491110','团员','健康','3','恩施','10086100861');
insert into StudentGerenInfor values('201717491101','群众','健康','4','十堰','10011100861');
insert into StudentGerenInfor values('202020491155','党员','健康','7','宣恩','10086100101');
insert into StudentGerenInfor values('201717491170','团员','健康','8','洛阳','10010100101');
insert into StudentGerenInfor values('201616491180','团员','健康','4','商丘','10011100811');
--录入学生的家庭信息
insert into SGuanxi values('202020491111','陈思','母亲','43','在家');
insert into SGuanxi values('202020491111','陈十','父亲','43','在家');
insert into SGuanxi values('201616491110','陈一','母亲','40','在家');
insert into SGuanxi values('201616491110','陈二','母亲','40','在家');
insert into SGuanxi values('201717491101','陈三','母亲','43','在家');
insert into SGuanxi values('201717491101','陈五','母亲','40','在家');
insert into SGuanxi values('202020491155','王一','母亲','40','在家');
insert into SGuanxi values('202020491155','王二','父亲','40','在家');
insert into SGuanxi values('202020491155','王三','姐姐','25','在家');
insert into SGuanxi values('201717491170','陈六','母亲','40','在家');
insert into SGuanxi values('201717491170','陈七','父亲','42','在家');
insert into SGuanxi values('201616491180','陈八','母亲','43','在家');
insert into SGuanxi values('201616491180','陈九','母亲','43','在家');
创建相应的触发器一共6种情况:
use SDbase
--触发器
--当学生入学成绩成绩小于460时候,学生的录取类型为借读生。
CREATE TRIGGER Student_fenshu ON StudentInfor --该触发器建在学生入学表生
FOR INSERT --触发器类型 插入记录时候
as
declare e1 cursor for
select Sno,enrollmentScore,muqianZhuangtai from inserted --触发器名称 e1
declare @Sno char(12)
declare @enrollmentScore smallint
declare @sourceUpdate char(10)
OPEN e1
fetch NEXT FROM e1 into @Sno,@enrollmentScore,@sourceUpdate
WHILE @@FETCH_STATUS=0 -- 当返回值为0时候继续执行Begin下语句,知道返回值为-1或者非0
BEGIN
IF @enrollmentScore <=460 --入过入学分数小于或者等于460
update StudentInfor set sourceUpdate='借读' --把当前已经输入的入读类型为借读,即使之前是其他也会更改为借读
where @Sno=Sno
fetch next from e1 into @Sno,@enrollmentScore,@sourceUpdate --条件为0时候继续遍历临时表中的记录,知道条件为非0
end
close e1 --遍历游标 e1
DEALLOCATE e1 --释放游标
--学籍管理,当学生毕业时候,在学籍变动表中,为学生发放毕业证
Create TRIGGER Student_biye ON xuejiUpdate
FOR insert --触发器类型 插入记录时候
as
declare e_biye cursor for
select Sno,updateType,SendOrYanZheng from inserted
declare @Sno varchar(36)
declare @updateType char(4)
declare @SendOrYanZheng VARCHAR(50)
OPEN e_biye
fetch NEXT FROM e_biye into @Sno,@updateType,@SendOrYanZheng
WHILE @@FETCH_STATUS=0
BEGIN
if @updateType='毕业' --如果学籍异动表中出现一个新的异动类型为毕业
BEGIN
update xuejiUpdate set SendOrYanZheng='发给毕业证' where @Sno=Sno --那么发给毕业证
END
fetch next from e_biye into @Sno,@updateType,@SendOrYanZheng
end
close e_biye
DEALLOCATE e_biye
--当学生状态为毕业状态时候,自动把所有毕业生的信息导入到一张毕业生表中去。
CREATE TRIGGER biye_tig ON StudentInfor
FOR update
as
declare e_insertBiye cursor for select Sno,Sname,buBie,sourceUpdate from inserted
declare @Sno char(12)
declare @Sname varchar(36)
declare @buBie char(4)
declare @sourceUpdate char(10)
OPEN e_insertBiye
fetch NEXT FROM e_insertBiye into @Sno,@Sname,@buBie,@sourceUpdate
WHILE @@FETCH_STATUS=0
BEGIN
--当我更改变动类型时候,而且满足变动类型为毕业,那么到处导出毕业生到BiyeSheng表中
if update(sourceUpdate) and @sourceUpdate='毕业'
insert into BiyeSheng values(@Sno,@Sname,@buBie)
fetch next from e_insertBiye into @Sno,@Sname,@buBie,@sourceUpdate
end
close e_insertBiye
DEALLOCATE e_insertBiye
--当学生学籍状态发生改变时候例如请假转班或者进行毕业,
--把学籍异动信息进行存储到学籍异动表中,每次变动都会添加一条记录,主键是学号+日期。
Create TRIGGER Student_baingeng ON StudentInfor
FOR update
as
declare e cursor for select Sno,muqianZhuangtai,sourceUpdate from inserted
declare @Sno varchar(36)
declare @muqianZhuangtai char(10)
declare @sourceUpdate char(10)
OPEN e
fetch NEXT FROM e into @Sno,@muqianZhuangtai,@sourceUpdate
while @@FETCH_STATUS=0
BEGIN
if update(sourceUpdate)
BEGIN
insert into xuejiUpdate values(@Sno,@sourceUpdate,getDate(),@muqianZhuangtai,'无')
END
fetch next from e into @Sno,@muqianZhuangtai,@sourceUpdate
end
close e
DEALLOCATE e
--当学校录取一名学生时候,为该生创建一个学籍管理系统的学生账户。
CREATE TRIGGER Student_Chufa ON StudentInfor
FOR INSERT
as
declare e cursor for select Sno,Sname from inserted
declare @Sno varchar(36)
declare @Sname varchar(36)
OPEN e
fetch NEXT FROM e into @Sno,@Sname
WHILE @@FETCH_STATUS=0
BEGIN
--创建账户,的登录账户为学号,姓名为学生姓名,权限代码为4,默认为学生
insert into UserType values(@Sno,123456,@Sname,4,'学生')
fetch next from e into @Sno,@Sname
end
close e
DEALLOCATE e
--创建一个新班级时候,为该班班主任老师授予查询该班学生信息的权利
CREATE TRIGGER Class_Table ON classTbale
FOR INSERT
as
declare e cursor for select TeacherNo,TeacherName from inserted
declare @TeacherNo varchar(12)
declare @TeacherName varchar(50)
OPEN e
fetch NEXT FROM e into @TeacherNo,@TeacherName
WHILE @@FETCH_STATUS=0
BEGIN
insert into UserType values(@TeacherNo,123456,@TeacherName,2,'老师')
fetch next from e into @TeacherNo,@TeacherName
end
close e
DEALLOCATE e
为了保持数据的完整性:
我们对相应的表进行了级联删除与级联更新的约束建立,该约束是列级约束,也可以在创建表的时候建立,本实例是创建表后建立的:
--对参照StudentInfor学生入学表的表建立级联更新与删除
alter table StudentGerenInfor add constraint a_Sno foreign key(Sno) references StudentInfor(Sno) on delete cascade
alter table SGuanxi add constraint b_Sno foreign key(Sno) references StudentInfor(Sno) on delete cascade
alter table xuejiUpdate add constraint c_Sno foreign key(Sno) references StudentInfor(Sno) on delete cascade