一、数据完整性的类型 空规则 空规则 (null rule)是定义在某一列上的规则,其作用是允许或禁止将要被插入或更新的数据行此列的为空(null),即没有 唯一列 唯一规则 (unique value rule)是定义在某一列(或某一列集)上的规则,其作用是确保将要被插入
一、数据完整性的类型
空规则
空规则(null rule)是定义在某一列上的规则,其作用是允许或禁止将要被插入或更新的数据行此列的值为空值(null),即没有值
唯一列值
唯一值规则(unique value rule)是定义在某一列(或某一列集)上的规则,其作用是确保将要被插入或更新的数据行此列(或列集)的值是唯一的。
主键值
主键值规则(primary key value rule)是定义在某一键(key)(键指一列或一个列集)上的规则,其作用是确保表内的每一数据行都可以由某一个键值唯一地确定。
引用完整性规则
引用完整性规则(referential integrity rule)是定义在某一键(key)(键指一列或一个列集)上的规则,其作用是确保任意键值都能与相关表(related table)的某一键值(即引用值(referenced value))相匹配
在引用完整性中还包含了如下规则:对引用值可以进行哪些类型的数据操作(data manipulation),以及这些操作将如何影响依赖值(dependent value)。引用完整性中包含的具体规则有:
复杂完整性检查
复杂完整性检查(complex integrity checking)是一种用户定义的规则,针对某一列(或某一列集),其作用是依据数据行的列值来允许或禁止插入,更新,或删除此数据行。
二、完整性约束描述
完整性约束(integrity constraint)指以显式声明的方式为数据表的列定义规则。Oracle 支持以下类型的完整性约束:
UNIQUE 键约束与 NOT NULL 约束结合使用
如 图 21-3 与 图 21-4 所示,定义了 UNIQUE 键约束的列可以输入空值(null),而同时定义了NOT NULL 约束的列则不能输入空值。由于空值与任何值比较都无意义,因此定义了 UNIQUE 键约束而没有定义 NOT NULL 约束的列上可以包含多行值为空的记录。多行列为空值(或复合UNIQUE 键(composite UNIQUE key)的所有列均为空值)不会违反 UNIQUE 键约束
引用完整性约束
在关系型数据库中,不同的表可以依据其共同的列产生关联关系,数据库需要确保数据遵从列关系的规则。引用完整性规则(referential integrity rule)就是用于确保列关系的规则。
以下是与引用完整性约束(referential integrity constraint)相关的术语。
|
|
术语 | 定义 |
|
|
外键(foreign key) | 引用完整性约束定义中包含的列(一列或多列),外键引用了引用键。 |
引用键(referenced key) | 被外键引用表(可以是不同的表或同一个表)的唯一键(unique key)或主键(primary key)。 |
依赖表(dependent table)或子表(child table) | 定义了外键约束的表。此表依赖于被引用的唯一键值或主键值。 |
引用表(referenced table)或父表(parent table) | 被子表外键引用的表。此表的引用键决定了能否向子表插入数据或更新其中的数据。 |
外键约束可以定义于多列上。但复合外键(composite foreign key)必须引用列数及列数据类型均相同的复合主键(composite primary key)或复合唯一键(composite unique key)。由于复合主键及复合唯一键的列数不能超过 32 个,因此复合外键也受此限制。
空值与外键
关系型模型允许外键(foreign key)与其引用的主键(primary key)或唯一键(unique key)相匹配,或为空值(null)。如果一个复合外键的部分列为空,则此外键的非空列值无需与父键(parent key)中对应的列值相匹配。
在引用完整性约束(referential integrity constraint)定义中,可以设定当被引用的父键(parent key)值被修改后,对子表(child table)中的依赖数据该执行何种操作。Oracle FOREIGN KEY 完整性约束定义中支持的引用操作(referential action)有UPDATE,DELETE NO ACTION,及DELETE CASCADE。
禁止操作
禁止操作(No Action)选项(默认操作)的含义是:如果对引用键值(referenced key value)的更新或删除操作将破坏引用完整性约束,则此操作不能执行。例如,当一个主键值被一个外键值引用时,这个主键值拥有依赖数据,所以不能被删除。
串联删除
串联删除(delete cascade)选项的含义是:如果包含引用键值(referenced key value)的数据行被删除,则子表(child table)中所有包含依赖外键值(dependent foreign key value)得数据行也将被删除。例如,当父表(parent table)的一行数据被删除时,如果此行的主键值被子表中一个或多个外键值引用,则子表中引用了此主键值的所有数据行也将被删除。
置空
置空(set null)选项的含义是:如果包含引用键值(referenced key value)的数据行被删除,则子表(child table)中所有依赖外键值(dependent foreign key value)将被置空。例如,在TMP 表中 employee_id 列引用了manager_id 列,删除一条经理数据时,所有为此经理工作的员工的 manager_id 值将被置空。
引用操作对 DML 的限制
采用不同引用操作(referential action)选项时,对父表(parent table)的主键值(primary key value)/唯一键值(unique key value)及子表(child table)的外键值(foreign key value)可以执行的 DML 语句。
|
||
DML 语句 |
对父表执行 |
对子表执行 |
|
||
INSERT | 当父键值唯一时允许执行。 | 当外键值能与一个父键值匹配,或外键值部分或全部为空时允许执行。 |
UPDATE No Action | 如果更新操作不会使子表内的任何数据行没有可引用的父键值,则允许执行。 | 如果新外键值依旧能与一个父键值匹配,则允许执行。 |
DELETE No Action | 如果子表没有引用此主键值则允许执行。 | 允许执行。 |
DELETE Cascade | 允许执行。 | 允许执行。 |
DELETE Set Null | 允许执行。 | 允许执行。 |
并发控制,索引,及外键
在大多数情况下,用户需要在外键(foreign key)上创建索引。但有一种情况例外,当外键所引用的主键(primary key)或唯一键(unique key)从不更新或删除时,外键上可以不创建索引。
Oracle 负责进行并发控制(concurrency control),确保数据访问时父键(parent key)及依赖外键(dependent foreign key)之间关系的正确性。用户可以根据实际情况选择并发控制的机制。以下各节将介绍不同情况,及在各种情况下 Oracle 建议的并发控制机制。
外键上无索引
如果外键(foreign key)上没有定义索引,当父表(parent table)中的数据行被更新或删除时,Oracle 采用的锁机制(locking mechanism)如图 21-8
所示。而向父表中插入数据时无需对子表(child table)加锁。
如果外键上没有定义索引,对父表的主键执行 DML 操作时必须获得子表上的共享行排他表级锁(share row exclusive table lock)(也称为share-subexclusive table lock,SSX)。此锁能够阻止其他事务对子表执行 DML 操作。SSX 锁在获得后立即释放。如果父表中有多个主键值被更新或删除,对每行执行 DML 操作时都需要对子表进行一次加锁及解锁操作。
如果外键上定义了索引,则对父表的主键执行 DML 操作时只需获得子表上的行共享表级锁(row share table lock)(也称为 subshare table lock,SS)。此锁不允许其他事务排他地对子表加锁,但允许对父表及子表执行 DML 操作。
如果子表的外键约束定义中指定了 ON DELETE CASCADE 选项,则删除主表数据时将导致相关的子表数据同时被删除。在这种情况下,Oracle 采用的锁机制与用户先手工地删除主表数据,再删除相关的子表数据时采用的锁机制相同。
CHECK 完整性约束
CHECK 完整性约束(integrity constraint)定义于列或列集上,此约束要求数据行满足用户定义的检查条件(或条件判断结果不确定(unknown))。如果一个 DML 语句使 CHECK 完整性约束的检查结果为假(false),则此语句将被回滚(rolled
back)。
检查条件
用户可以使用 CHECK 约束定义检查条件(check condition)来实现特殊的完整性规则(integrity rule)。定义CHECK 约束的检查条件时有以下限制
同一列可以被多个
CHECK 约束(constraint)的条件定义所引用。用户为某一列定义的CHECK 约束的数量不受限制。
如果用户为在一列上创建了多个 CHECK 约束,必须确保各个约束的检查条件不会相互冲突。CHECK 约束的检查顺序是不确定的。Oracle 也不会检查各个CHECK 约束是否为互斥的(mutually exclusive)。
延迟约束检查
用户可以将约束检查(checking constraint for validity)延迟(defer)至事务结束时进行。
约束属性
用户可以将约束定义为可延迟的(deferrable)或不可延迟的(not deferrable),以及初始为延迟的(initially deferred)或初始为即时的(initially immediate)。上述属性应与不同类型的约束结合使用。用户可以在CONSTRAINT 子句中使用以下关键字进行设定:
约束状态
此外: