一、数据完整性的类型
关系模型的三类完整性是实体完整性、参照完整性和用户完整性。
实体完整性和参照完整性是关系模型必须满足的完整性约束条件,应由关系系统自动支持。
1、实体完整性 (entity integrity)
实体完整性是基于主码的,一个主码由一个或多个属性组成。实体完整性要求主码中的任一属性(列)不能为空。之所以要保证实体完整性主要是因为:在关系中,每一个元组的区分是依据主码值的不同,若主码值取空值,则不能标明该元组的存在。
2、参照完整性 (referential integrity)
参照完整性是基于外码的,若关系R中含有与另一关系S的主码PK相对应的属性组FK(FK称为R的外码),则参照完整性要求,对R中的每个元组在FK上的值必须是S中某个元组的PK值,或者为空值。
参照完整性的合理性在于:R中的外码只能对S中主码的引用,不能是S中主码没有的值.如学生和选课表两关系,选课表中的学号是外码,它是学生表的主键,若选课表中出现了某个学生表中没有的学号,即某个学生还没有注册,却已有了选课记录,这显然是不合理的。
3、用户定义的完整性(user-defined integrity)
实体完整性和参照完整性适用于任何关系数据库系统。除此之外,不同的关系数据库系统根据其应用环境的不同,往往还需要一些特殊的约束条件。用户定义的完整性就是针对某一具体关系数据库的约束条件,它反映某一具体应用所涉及的数据必须满足的语义要求。如:学生的成绩一般情况下的取值范围在0-100之间。
二、 Oracle的数据完整性的实现
1、 Oracle中的实体完整性
实体完整性规则要求主属性非空。 Oracle在CREATE TABLE语句中提供了PRIMARY KEY子句,供用户在建表时指定关系的主码列。例如:在学生选课数据库中,要定义Student表的sno属性为主码,可使用如下语句:
SQL>CREATE TABLE Student
( sno NUMBER(8),
sname VARCHAR(20),
sage NUMBER(20),CONSTRAINT PK_SNO PRIMARY KEY (sno));
其中:PRIMARY KEY(SNO)表示SNO是Student表的主码。PK_SNO是此主码约束名。
在用PRIMARY KEY语句定义了关系的主码后,每当用户程序对主码列进行更新操作时,系统自动进行完整性检查,凡操作使主码值为空值或使主码值在表中不唯一,系统拒绝此操作,从而保证了实体完整性。
2、ORACLE中的参照完整性
Oracle的CREATE TABLE语句不仅可以定义关系的实体完整性规则,也可以定义参照完整性规则,即用户可以在建表时用FOREIGN KEY 子句定义哪些列为外码列,用REFERENCES子句指明这些外码相应于哪个表的主码,用ON DELETE CASCADE 子句指明在删除被参照关系的元组时,同时删除参照关系中外码值等于被删除的被参照关系的元组中主码值中的元组。
SQL>CREATE TABLE SC
(sno NUMBER(8);
cno NUMBER(2),
grade NUMBER(3),
CONSTRAINT PK_SC PRIMARY KEY(sno,cno),CONSTRAINT FK_SNO FOREIGN KEY (sno) REFERENCES student(sno) on delete CASCADE);
3、ORACLE中的用户自定义完整性
除实体完整性和参照完整性外、
应用系统中往往还需要定义与应用有关的完整性约束。
例如:要求某一列的值能取空值;某一列的值在表中是唯一的;某一列的值要在某个范围中等。ORACLE允许用户在建表时定义下列完整性约束。
A、例如:部门表中保证部门名称唯一
SQL>CREATE TABLE DEPT
(deptno NUMBER,
dname VARCHAR(9) CONSTRAINT U1 unique,
loc VARCHAR(10),
CONSTRAINT PK_DEPT PRIMARY KEY (deptno));
B、例如:学生成绩大于0小于等于100(检查列值是否满足一个布尔表达式)。
SQL>CREATE TABLE SC
(sno NUMBER(8);
cno NUMBER(2),
grade NUMBER(3) CONSTRAINT C1 CHECK (0CONSTRAINT PK_SC PRIMARY KEY(sno,cno),CONSTRAINT FK_SNO FOREIGN KEY (sno) REFERENCES student(sno) on delete CASCADE);
在ORACLE中,用户还可以通过触发器定义其他完整性约束。
如:在学生成绩表中定义如下完整性规则,“学生成绩低于60分,自动改为60分。“
CREATE TRIGGER UPDATE_GRADE
BEFORE INSERT OR UPDATE OF grade on sc
FOR EACH ROW
BEGIN
IF :new.grade<60 THEN
:new.grade=60;
END;