数据库的完整性是指数据的正确性和相容性。
数据库管理系统必须提供以下功能:
关系模型的实体完整性在create table
中用primary key
定义。
对于单一属性构成的码,可以定义为列级约束条件或者表级约束条件;
对于多个属性构成的码,只能采用表级定义(详见样例):
create table Student(
Sno char(9) primary key,
/*在列级定义主码*/
);
create table Student(
Sno char(9),
primary key(Sno)
/*在表级定义主码*/
);
create table SC(
Sno char(9) not null,
Cno char(4) not null,
primary key(Sno, Cno)
/*只能在表级定义主码*/
)
在定义了表的主码之后,每当插入新的记录或是修改记录的时候,都会进行如下的检查:
关系模型的参照完整性在create table
中使用foreign key
定义哪些为外码,用references
短语指明这些外码参照哪些表的主码。
样例:
create table SC(
Sno char(9) not null,
Cno char(4) not null,
Grade smallint,
primary key(Sno, Cno),
foreign key(Sno) references Student(Sno),
foreign key(Cno) references Course(Cno),
);
对被参照表和参照表进行增删修改有可能会破坏参照完整性,必须进行检查以确保两个表的相容性。
被参照表(如Student ) |
参照表(如SC ) |
违约处理 |
---|---|---|
参照完整性被破坏 | 插入元组 | 拒绝 |
参照完整性被破坏 | 修改外码值 | 拒绝 |
删除元组 | 参照完整性被破坏 | 拒绝、级联删除、设为空值 |
修改主码值 | 参照完整性被破坏 | 拒绝、级联删除、设为空值 |
解释:
SC
中增加或修改一个元组,该元组的Sno
在Student
找不到对应的值。Student
中删除或修改一个元组,造成在SC
表中有些Sno
在Student
中找不到对应的值。上述情况发生时,一般有两种解决方案:
no action
,此为**默认策略*。cascade
,删除或修改参照表中所有导致不一致的元组。显示说明参照完整性的违约处理示例:
create table SC(
Sno char(9),
Cno char(4),
Grade smallint,
/*表级实体完整性,Sno和Cno都不能为空*/
primary key(Sno, Cno),
/*表级定义参照完整性*/
foreign key(Sno) references Student(Sno)
on delete cascade
on update cascade,
foreign key(Cno) references Course(Cno)
on delete no action
on update cascade
);
用户定义的完整性就是指针对某一具体应用的数据必须满足的语义要求。
在create table
中定义属性的同时,可以同时定义约束条件,包括:
not null
unique
check
语句create table Student(
Sno char(9) unique not null,
Sname char(8) not null,
Ssex char(2) check (Ssex in ('男', '女')),
);
在插入新的元组或者修改元组时,如果不满足属性上的约束条件,操作将被拒绝执行。
在create table
的同时可以使用check
语句定义元组级别的约束条件(即不同的属性之间应该满足的条件)。
/*当性别是男时,姓名不能以Ms.开头(什么鬼样例)*/
create table Student(
Sname char(8) not null,
Ssex char(2) not null,
check(Ssex == '女' or Sname not like 'Ms.%')
);
当插入或是修改元组的时候,元组上的约束条件将被检查,不满足的操作将会被拒绝执行。
SQL在create table
中提供了完整性约束命名子句constraint
,可以更加灵活的删除、增加一个约束条件。
完整性约束命名子句
constraint <完整性约束条件名> <完整性约束条件>;
create table Student(
Sno numeric(6)
constraint c1 check(Sno between 100000 and 999999),
Sname char(20)
constraint c2 not null,
Sage numeric(3)
constraint c3 check(Sage < 30),
Ssex char(2)
constraint c4 check(Ssex in ('男', '女')),
constraint SK primary key(Sno),
);
修改表中的完整性限制
可以用alter table
修改。
alter table Student
drop constraint c3;
alter table Student
add constraint cc check(Sagr < 40);
通过使用create assertion
语句来创建断言,可以用来指定更具一般性的约束,可以定义涉及多个表或聚集操作的比较复杂的完整性约束。
在断言创建之后,任何对断言中涉及的操作都会出发关系数据库管理系统对断言的检查,不能使断言为真的操作都会被拒绝执行。
创建断言的语句格式
create assertion <断言名> <check 子句>;
样例:限制每门课最多60人选课。
create assertion sample1
check (60 >= all(select count(*) from SC group by Cno));
删除断言的语句格式
drop assertion <断言名>;
定义:用户定义在关系表上的一类由事件驱动的特殊过程。
触发器又叫做事件-条件-动作规则,当特定的系统事件发生时,对规则的条件进行检查,如果成立则执行动作。
建立触发器的一般格式:
create trigger <触发器名>
{before | after} <触发事件> on <表名>
referencing new|old row as <变量>
for each {row | statement}
[when <触发条件>] <触发动作体>
详细说明:
样例:单独记录成绩增加超过10%的学生:
create trigger SC_T
after update of Grade on SC
referencing
oldrow as oldtuple,
newrow as newtuple
for each row
when (newtuple.Grade >= oldtuple.Grade * 1.1)
insert into SC_T (Sno, Cno)
values (oldyuple.Sno, oldtuple.Cno);
触发器激活时的执行顺序:
before
触发器after
触发器如果存在多个before|after
触发器,遵循先创建先执行的原则。
drop trigger <触发器名> on <表名>;