实体完整性和参照完整性是关系模型必须满足的完整性约束条件,应由关系系统自动支持。
1、实体完整性
(entity
integity)
实体完整性是基于主码的,一个主码由一个或多个属性组成。实体完整性要求主码中的任一属性(列)不能为空,所谓空值
是“不知道”或“无意义”的值。之所以要保证实体完整性主要是因为:在关系中,每一个元组的区分是依据主码值的不同,若主码值取空值,则不能标明该元组的
存在。
2、参照完整性
(efeential
integity)
参照完整性是基于外码的,若基本关系R中含有与另一基本关系S的主码PK相对应的属性组FK(FK称为R的外码),则参照完整性要求,对R中的每个元组在FK上的值必须是S中某个元组的PK值,或者为空值。
参照完整性的合理性在于:R中的外码只能对S中主码的引用,不能是S中主码没有的值如学生和选课表两关系,选课表中的学号是外码,它是学生表的主键,若选课表中出现了某个学生表中没有的学号,即某个学生还没有注册,却已有了选课记录,这显然是不合理的。
3、用户定义的完整性(use-defined
integity)
实体完整性和参照完整性适用于任何关系数据库系统。除此之外,不同的关系数据库系统根据其应用环境的不同,往往还需
要一些特殊的约束条件。用户定义的完整性就是针对某一具体关系数据库的约束条件,它反映某一具体应用所涉及的数据必须满足的语义要求。如:学生的成绩一般
情况下的取值范围在0-100之间。
二、Oacle的数据完整性的实现
1、Oacle中的实体完整性
实体完整性规则要求主属性非空。Oacle在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中的参照完整性
Oacle的CREATE
TABLE语句不仅可以定义关系的实体完整性规则,也可以定义参照完整性规则,即用户可以在建表时用FOREIGN
KEY
子句定义哪些列为外码列,用REFERENCES子句指明这些外码相应于哪个表的主码,用ON
DELETE
CASCADE
子句指明在删除被参照关系的元组时,同时删除参照关系中外码值等于被删除的被参照关系的元组中主码值中的元组。
SQL
CREATE
TABLE
SC
(sno
NUMBER
(8);
cno
NUMBER
(2),
g
ade
NUMBER
(3),
CONSTRAINT
PK_SC
PRIMARY
KEY(sno,cno),CONSTRAINT
FK_SNO
FOREIGN
KEY
(sno)
REFERENCES
student(sno)
on
delete
CASCADE);
建立外键?
例:学生表student (id, name , sex )
成绩表score (id ,math )
如何创建表,要求 有主键,有约束 解:
CREATE TABLE STUDENT(ID CHAR(10), NAME VARCHAR(8),SEX CHAR(1));
ALTER TABLE STUDENT ADD CONSTRAINT PK_STUDENT PRIMARY KEY(ID);
CREATE TABLE SCORE( ID CHAR(10),MATH NUMBER(5,2));
ALTER TABLE SCORE ADD CONSTRAINT FK_SCROE FOREIGN KEY(ID) REFERENCES STUDENT(ID);
主键与外键:
键是表中的列(可以是一列,也可以是几列),主键用于唯一的标识表中的数据项;外键用于连接父表和子表。而所谓的父表和子表是根据3NF
范式的要求,为了消除传递依赖,将原表拆成2个相互关联的表,而这个关联就是外键。
特总结了Oracle和DB2数据库下如何禁用外键约束的方法。
一、Oracle数据库:
禁用约束基本语法:
alter table 数据库表名 disable constraint 约束名
假设现在需要关闭pub_organ的外键约束:
1、 首先查询pub_organ存在哪些外键约束,此时需要用到oracle的字典表user_constraints。
select from user_constraints where table_name='PUB_ORGAN';
上图就是查询结果,其中各字段含义如下:
OWNER: 表的所有者
CONSTRAINT_NAME: 约束名称
CONSTRAINT_TYPE: 约束类型(R代表外键,P代表主键,C代表check约束)
TABLE_NAME: 表名称
SEARCH_CONDITION: check约束的具体信息
STATUS: ENABLED表示当前约束是启用的,DISABLED表示当前约束未启用。
2、 查询出表存在哪些约束后,即可以通过alter语句启用或禁用指定的约束了。
如禁用pub_organ表的外键PUBORGAN_FK1,则可以使用如下命令实现:
alter table PUB_ORGAN disable constraint PUBORGAN_FK1;
执行后,再次查询字典表user_constraints,如下:
此时往数据库表pub_organ中插入数据时就不再受外键约束的影响了。
启用约束基本语法:
alter table 数据库表名 enable constraint 约束名
如现在需要重新启用pub_organ的外键约束,可以使用如下命令:
alter table PUB_ORGAN enable constraint PUBORGAN_FK1;
二、DB2数据库:
禁用约束基本语法:
ALTER TABLE 表名称 ALTER FOREIGN KEY 约束名称 NOT ENFORCED
启用约束基本语法:
ALTER TABLE 表名称 ALTER FOREIGN KEY 约束名称 ENFORCED
相关字典表:SYSIBMSYSTABCONST
如:select from SYSIBMSYSTABCONST where tbname='PUB_ORGAN';
各字段含义如下:
NAME: 约束名称
DEFINER: 定义者
CONSTRAINTTYP: 约束类型(P代表主键,F代表外键)
TBNAME: 表名称
ENFORCED: 是否启用(Y代表启用,N代表未启用)
三、封装成java接口、批量执行
在实际工作中,经常会将若干个表,或者所有数据库表的外键一起禁用,此时需要批量执行相关命令,笔者根据工作实际,使用java封装了相关接口,以方便使用。
对外暴露接口如下:
/
启用当前用户指定tableName的所有外键约束
入参使用可变参数(jdk5新特性)
调用方式:
1、enableFK("pub_organ")
2、enableFK("pub_organ","pub_stru")
3、enableFK(new String[]{"pub_organ","pub_stru"})
/
public static void enableFK(StringtableNames){
disableORenbaleFK(true,tableNames);
}
/
禁用当前用户指定tableName的所有外键约束
/
public static void disableFK(StringtableNames){
disableORenbaleFK(false,tableNames);
}
/
启用当前用户所有表的外键约束
/
public static void enableAllFK(){
disableORenableAllConstraint(true);
}
/
禁用当前用户所有表的外键约束
/
public static void disableAllFK(){
disableORenableAllConstraint(false);
}
其中核心处理代码如下:
if(tableNames==null||tableNameslength==0){
throw new RuntimeException("入参tableNames不能为空!");
}
//查询指定表的外键约束
String sql = null;
String dbType = getDBType();
if(dbTypecontains("ORACLE")){
sql = "select 'alter table ' || table_name || ' disable constraint ' || constraint_name from user_constraints where constraint_type='R' and TABLE_NAME in(";
if(isEnable){
sql = sqlreplace("disable", "enable");
}
}else if(dbTypecontains("DB2")){
sql = "select 'ALTER TABLE ' || TBNAME || ' ALTER FOREIGN KEY ' || NAME ||' NOT ENFORCED ' FROM SYSIBMSYSTABCONST WHERE CONSTRAINTYP='F' and TBNAME in(";
if(isEnable){
sql = sqlreplace("NOT ENFORCED", "ENFORCED");
}
}else{
throw new RuntimeException("数据库类型无效(仅支持Oracle和DB2),dbType="+dbType);
}
StringBuffer generateSQL = new StringBuffer(sql);
for(int i=0;i<tableNameslength;i++){
generateSQLappend(" '");
generateSQLappend(tableNames[i]toUpperCase());//注意须转换成大写
generateSQLappend("',");
}
generateSQLdeleteCharAt(generateSQLlength()-1);
generateSQLappend(")");
List<Map<String, Object>> dataSet = DBToolexecuteQuery(generateSQLtoString());
//启用or停止查询出的外键约束
for(int i=0;i<dataSetsize();i++){
Map<String, Object> record = dataSetget(i);
Iterator<Entry<String, Object>> itor = recordentrySet()iterator();
while(itorhasNext())
{
Entry<String, Object> e = itornext();
DBToolexecuteUpdate(egetValue()toString(),UpdateTypeALTER);
}
}
以上就是关于Oracle的数据完整性有哪些类型全部的内容,包括:Oracle的数据完整性有哪些类型、oracle数据库中表跟表怎样关联、如何启用和禁用oracle&DB2数据库外键约束等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)