版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、SQLServer - 约束一、约束的分类在SQLServer中,有3种不同类型的约束。1、实体约束实体约束是关于行的,比如某一行出现的值就不允许出现在其他行,例如主键。2、域约束域约束是关于列的,对于所有行,某一列有那些约束,例如CHECK约束。3、参照完整性约束如果某列的值必须与其他列的值匹配,那就意味着需要一个参照完整性约束,例如外键。二、约束命名在学习约束之前,首先来了解下为约束命名需要注意哪些地方。SQLServer在我们不提供名称时,会自动创建名称,但是由系统自动创建的名称并不是特别有用。例如,系统生成的主键名称可能是这样的:PK_Employees_145C0A3F。
2、PK代表主键(primary key),Employees代表在Employees表中,而剩下的“145C0A3F”部分是为了保证唯一性而随机生成的值。只有通过脚本创建才会得到这种值,如果是通过Managerment Studio创建表,那么就直接是PK_Employees。对于系统自动生成的Check约束名称如:CK_Customers_22AA2996。CK代表这是一个Check约束,Customers代表是在Customers表中,后面的22AA2996还是一个随机数。如果一个表中有多个Check约束,则命名可能如下:CK_Customers_22AA2996CK_Customers_2
3、5869641CK_Customers_267ABA7A如果你需要修改这些约束其中的一个,那么你很难分辨这些约束到底是哪一个。因此,为了能够一眼看上去就知道这个约束是用来干什么的,我们应该使用一种简单明了的短语来进行命名。例如要确保某一列电话号码格式正确的约束,我们可以使用命名CK_Customers_PhoneNo这样的短语来命名。总之命名要做到以下几点:1、一致性2、通俗易懂3、满足以上两个条件的情况下简化名称。三、键约束1、主键约束 主键是每行的唯一标识符,仅仅通过它就能准确定位到一行,其中主键列在整个表中不能有重复,必须包含唯一的值(不能为NULL)。由于主键在关系数据库中的
4、重要性,因此它是所有键和约束中最重要的。 下面来说说主键的创建方式1、在创建表的时候创建主键约束。create table customer( customerId int identity not null primary key,-创建主键约束 CustomerName nvarchar(30) not null);怎么样,非常简单吧!2、在已存在的表上创建主键约束现在假设已经存在了一张表,但是还没有主键约束:alter table person add constraint PK_Employee_Id-外键名称 primary key(personId)-personId 字
5、段名alter名称告诉SQLServer如下信息:1、添加了一些内容到表中(也可以删除表中的某些内容)2、添加了什么内容(一个约束)3、对约束的命名(允许以后直接访问约束)4、约束的类型(主键约束)5、约束应用于哪个列。3、复合主键的创建如果实在Management Studio中,创建复合主键,只需要按住Ctrl键,选中两个列,然后设置为主键就OK了,非常简单。下面主要讲述使用T-SQL创建复合主键的方法:ALTER TABLE 表名 WITH NOCHECK ADD CONSTRAINT PK_表名 PRIMARY KEY NONCLUSTERED ( 字段名1, 字段名2 ) 在多对多联
6、系中,常常会有一张表来描述其他两张表的关系,就以此读者和书为例子:ALTER TABLE ReaderAndBook ADD CONSTRAINT PK_ReaderAndBook PRIMARY KEY NONCLUSTERED ( ReaderId, BookId ) 2、外键约束外键既能确保数据完整性,也能表现表之间的关系。添加了外键之后,插入引用表的记录要么必须被引用表中被引用列的某条记录匹配,要么外键列的值必须设置为NULL。外键和主键不一样,每个表中的外键数目不限制唯一性。在每个表中,每一有-253个外键。唯一的限制是一个列只能引用一个外键。一个列可以被多个外键引用。1、创建表的时
7、候创建外键create table orders( orderId int identity not null primary key, customerId int not null foreign key references customer(customerId)-约束类型-外键-引用表(列名);2、在已存在的表中添加一个外键假设上面的代码去掉了添加外键行,那么可以书写代码如下:alter table orders add constraint FK_Orders_CustomerId -添加约束 名称 foreign key (customerId) references custo
8、mer(customerId) -外键约束,外键列名,被引用列名刚添加的约束和之前添加的约束一样生效,如果某行引用customerId不存在,那么就不允许把该行添加到Orders表中。3、级联动作外键和其他类型键的一个重要区别是:外键是双向的,即不仅是限制子表的值必须存在于父表中,还在每次对父表操作后检查子行(这样避免了孤行)。SQLServer的默认行为是在子行存在时“限制”父行被删除。然而,有时会自动删除任何依赖的记录,而不是防止删除被引用的记录。同样在更新记录时,可能希望依赖的记录自动引用刚刚更新的记录。比较少见的情况是,你可能希望将引用行改变为某个已知的状态。为此,可以选择将依赖行的值
9、设置为NULL或者那个列的默认值。这种进行自动删除和自动更新的过程称为级联。这种过程,特别是删除过程,可以经过几层的以来关系(一条记录依赖于另一条记录,而这另一条记录又依赖其他记录)。在SQLServer中实现级联动作需要做的就是修改外键语法-只需要在添加前面加上ON子句。例如:alter table orders add constraint FK_Orders_CustomerId -添加约束 名称 foreign key (customerId) references customer(customerId) -外键约束,外键列名,被引用列名 on update no action -默
10、认修改时不级联更新子表 on delete cascade -删除时级联删除依赖行当在进行级联删除时,如果一个表级联了另一个表,而另一个表又级联了其他表,这种级联会一直下去,不受限制,这其实是级联的一个危险之处,很容易一个不小心删掉大量数据。级联动作除了no action,cascade之外,还有set null和set default。后两个是在SQLServer2005中引入的,如果要兼容到SQLServer2000的话,要避免使用这两个级联动作。但是他们的才做是非常简单的:如果执行更新而改变了一个父行的值,那么子行的值将被设置为NULL,或者设置为该列的默认值(不管SET NULL还是S
11、ET DEFAULT)。4、外键其他方面的考虑外键中的之只有相中可能的选择:1、在列中填充与被引用表中的相应列相匹配的值。通过定义引用列为NOT NULL,可以使外键完全是必须的(即用户添加数据时必须引用表中必须有相匹配的一行数据)。2、不填充任何值,而使该值为NULL。允许引用列有NULL值时,用户可以选择不提供值-即使在被引用表没有与NULL值匹配的行,还是允许插入。3、唯一约束唯一约束与主键比较相似,共同点在于它们都要求表中指定的列(或者列的组合)上有一个唯一值,区别是唯一约束没有被看作表中记录的唯一标识符(即使你可以按这样的方式使用也有效),而且可以有多个唯一约束(而在每个表中只能有一
12、个主键)。一旦建立了唯一约束,那么指定列中的每个值必须是唯一的。如果更新或者插入一条记录在带唯一约束的列上有已经存在的值的记录,SQLServer将抛出错误,拒绝这个记录。和主键不同,唯一约束不会自动防止设置一个NULL值,是否允许为NULL由表中相应列的NULL选项的设置决定,但即使确实允许NULL值,一张表中也只能够插入一个NULL值(如果允许多个,那就不叫唯一了)。在已存在的表上创建唯一约束:alter table Account add constraint AK_AccountName -约束名 unique (Account_Name) - 列名AK代表替换键(Alternate
13、Key),唯一约束也叫替换键。主键和唯一约束的区别:· 主键约束不允许出现NULL值。任何索引的索引键都不允许包含null值。但唯一约束允许包含NULL值,但唯一约束把两个NULL值当作重复值,所以施加了唯一约束的每一列只允许包含一个NULL值。· 创建主键时会自动创建聚集索引,除非当前表中已经含有了聚集索引或是创建主键时指定了NONCLUSTERED关键字。· 创建唯一约束时会自动创建非聚集索引,除非你指定了CLUSTERED关键字并且当前表中还没有
14、聚集索引。· 每个表中只能有一个主键,但可以由多个唯一约束。4、CHECK约束CHECK约束约束可以和一个列关联,也可以和一个表关联,因为它们可以检查一个列的值相对于另外一个列的值,只要这些列都在同一个表中以及值是在更新或者插入的同一行中。CHECK约束还可以用于检查列值组合是否满足某一个标准。可以像使用where子句一样的规则来定义CHECK约束。CHECK约束条件的示例如下:目标SQL限制Month列为合适的数字BETWEEN 1 AND 12正确的SSN格式LIKE'0-90-90-9-0-90-9-0-90-90-90-9'
15、;限制为一个快递公司的特定列表IN('UPS','Fed Ex',EMS')价格必须为正数UnitPrice >= 0引用同一行中的另外一列ShipDate >= OrderDate上面给出的列表只是一小部分,而条件实际上市无限多的。几乎所有可以放到where子句的条件都可以放到该约束中。而且和其他选择(规则和触发器)相比,CHECK约束执行速度更快。在已存在的表中添加一个CHECK约束:alter table Account add constraint CN_AccountAge check (Account_Age > 18);
16、- 插入年龄必须大于18如果此时视图添加一条不满足的记录,将报如下错误:insert into Account values (22,'洪',17)消息 547,级别 16,状态 0,第 1 行INSERT 语句与 CHECK 约束"CN_AccountAge"冲突。该冲突发生于数据库"Nx",表"dbo.Account", column 'Account_Age'。语句已终止。5、DEFAULT约束和所有约束一样,DEFAULT约束也是表定义的一个组成部分,它定义了当插入的新行对于定义了默认约束的列未
17、提供相应数据时该怎么办。可以定义它为一个字面值(例如,设置默认薪水为0,或者设置字符串列为"UNKNOWN"),或者某个系统值(getdate()。对于DEFAULT约束,要了解以下几个特性:1、默认值只在insert语句中使用-在update语句和delete语句中被忽略。2、如果在insert语句中提供了任意值,那就不使用默认值。3、如果没有提供值,那么总是使用默认值。值得注意的是,update命令的规则由一个例外,如果显示说明使用默认值就是例外。可以通过使用关键字DEFAULT表示更新的值设置为默认值。5.1在创建表时定义DEFAULT约束:create table
18、person( person_id int identity not null primary key, person_name nvarchar(30) not null default '无名氏', person_age int not null)在执行语句后:insert into person (person_age) values(24)表中被插入一条记录如下: 5.2在已存在的表上添加DEFAULT约束:alter table person add constraint CN_DefaultName default '无名氏' for p
19、erson_name6、禁用约束 有时我们想暂时或永久地消除约束。但是SQL Server并没有提供删除约束的方法。SQL Server只允许禁用外键约束或CHECK约束,而同时保持约束的完整性。 禁用一个数据完整性规则通常是因为已经有无效数据了。这样的数据通常分为以下两类:1、在创建约束时已经在数据库中的数据2、在约束创建以后希望添加的数据SQL Server允许禁用完整性检查一段时间来对例外的无效数据作处理,然后再重新启用完整性(不是物理删除数据完整性约束)。注意:不能禁用主键约束或者唯一约束6.1、在创建约束时,忽略检查之前的不满足数据要添加一个约束,但是有不应用到已
20、存在的数据中,可以再执行Alter Table语句添加约束时使用WITH NOCHECK选项。按照上面创建Check约束的方法,已经Alter Table时,表中本身已经存在不符合的数据,那么Alter Table操作将被SQL Server拒绝执行。除非已经存在的所有数据都满足CHECK约束的条件,否则SQL Server不会执行创建约束的命令。要解决这个问题,我们可以添加WITH NOCHECK。我们先新建一个表只有3个字段的表,Id、姓名、年龄,并在里面插入一条不满足要求的数据:insert into Account values (23,'洪',17) 然后执
21、行添加约束命令:alter table Account add constraint CN_AccountAge18 check (Account_Age > 18); - 插入年龄必须大于18 SQL Server报一下错误:消息 547,级别 16,状态 0,第 1 行ALTER TABLE 语句与 CHECK 约束"CN_AccountAge18"冲突。该冲突发生于数据库"Nx",表"dbo.Account", column 'Account_Age'。 这时候我们换一种方式去执行:al
22、ter table Account WITH NOCHECK add constraint CN_AccountAge18 check (Account_Age > 18); - 插入年龄必须大于18 以上代码就能够成功执行,并且只有以后添加的数据具备约束,之前添加的不符合条件的数据记录依然存在。 6.2临时禁用已存在的约束 当我们需要从另一数据库中导入数据到表中,而表中已建立了约束的时候,可能会存在一些数据和规则不匹配。当然有一个解决方式是先删除约束,添加需要的数据,然后WITH NOCHECK在添加回去。但是这样做太麻烦了。我们不需要这么做。我们可以采用
23、名为NOCHECK的选项来运行ALTER语句,这样就能够取消需要的约束。先来看看上节中创建的这个约束:alter table Account add constraint CN_AccountAge18 check (Account_Age > 18); - 插入年龄必须大于18 要取消以上约束可以这样来:Alter Table Account NOCHECK constraint CN_AccountAge18执行命令:insert into Account values (25,'取消了约束',17)执行成功,成功添加了一行数据。留意到又能够向表中插入格式不
24、匹配的数据了。这里要说明下,如下知道一个约束是否是启用还是禁用呢?sp_helpconstraint命令,当我们执行sp_helpconstraint的时候,会有一列status_enabled显示该约束的启用状态:sp_helpconstraint Account 留意到status_enabled列为Disabled说明是禁用的意思。当要启用约束时,只需要用将语句中的NO CHECK替换为CHECK就可以了:Alter Table Account CHECK constraint CN_AccountAge18 执行之后,约束又启用了:再来sp_helpconstrai
25、nt看下: 留意到status_enabled列变成了Enabled。status_enabled的两种状态如下:Enabled:启用;Disabled:禁用;7、规则和默认值规则和默认值的应用要早于CHECK和DEFAULT约束。他们是较老的SQL Server备用约束的一部分,当然也不是没有优点。自7.0版本之后,MicroSoft列出规则和默认值只是为了向后兼容,而不准备在以后继续支持这个特性。因此对于生成新代码时,应该使用约束。规则、默认值与约束的本质区别是:约束是一个表的特征,本身没有存在形式,而规则和默认值是表和自身的实际对象,本身存在。约束是在表定义中定义的,而规则和默
26、认值是单独定义,然后"绑定到"表上。规则和默认值的独立对象特性使得它们可以在重用时不用重新定义。实际上,规则和默认值不限于被绑定到表上,它们也可以绑定到数据类型上。7.1规则 规则和CHECK约束非常相似。它们之间的唯一区别是规则每次只能作用于一个列。可以将同一规则分别绑定到一个表中的多个列,但是规则分别作用于每个列,根本不会意识到其他列的存在。像QtyShipped<= QtyOrdered这样的约束不适用于规则(它引用多个列),而LIKE(0-90-90-9)这样的定义适用于规则。定义规则:下面定义一个规则,这样就可以首先看到区别所在:CREATE RU
27、LE Age18Rule AS Age > 18这里比较的是一个变量,不管被检查的列是什么值,这个值将用于替换Age。因此在这个示例中,规则所绑定的任何列的值都必须大于18。到目前为止,只是创建了一个规则,但这个规则还没对任何表的任何列起作用,要激活这个规则需要使用一个存储过程:sp_bindrule。将规则Age18绑定到表person的person_age列:EXEC sp_bindrule 'Age18Rule','person.person_age'此时,如果我们执行不满足规则的插入操作:insert into person values (
28、9;绑定规则',17) 将返回如下报错信息:消息 513,级别 16,状态 0,第 1 行列的插入或更新与先前的 CREATE RULE 语句所指定的规则发生冲突。该语句已终止。冲突发生于数据库 'Nx',表 'dbo.person',列 'person_age'。语句已终止。很明显,规则已经生效。要特别注意的是,在绑定之前,规则与任何表,任何列都没有关系,因此在绑定的时候,第二个参数要加.指定表名与列名(tablename.column)。解除绑定规则:当我们需要在一个列上解除绑定规则的时候,只要执行sp_unbindrule
29、删除刚才绑定的规则:EXEC sp_unbindrule 'person.person_age'这时候,执行刚才的插入操作,就不会报错了。删除规则:如果希望将规则从数据库中彻底删除,那么可以在表中使用非常熟悉的DROP语法。DROP RULE <rule name>如删除刚才创建的那条规则:DROP RULE Age18Rule7.2默认值默认值类似于DEFAULT。实际上默认值-DEFAULT约束的关系与规则-CHECK约束的关系差不多。区别在于它们被追加到表中的方式和对用户自定义数据类型的默认值(是对象,而不是约束)支持。定义默认值的语法和定义规则类似:CREATE DEFAULT <default_name>AS <default value>创建默认值:因此,假设要为Age定义一个值为0的默认值:CREATE D
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 昆明云南昆明市五华区人民政府护国街道办事处招聘6名公益性岗位人员笔试历年参考题库附带答案详解
- 2024新能源电动车制造销售合同
- 2025年沪教版必修3英语上册阶段测试试卷
- 2025年新科版必修1历史下册阶段测试试卷含答案
- 2025年沪教版九年级地理上册月考试卷
- 2025年粤教沪科版高二历史下册月考试卷含答案
- 2025版文创产品设计委托合同模板协议3篇
- 2025年粤教版九年级历史上册阶段测试试卷
- 二零二五年度新能源汽车电池研发与制造合同3篇
- 2025年沪教版八年级地理下册月考试卷
- 血透室护士长述职
- 2024年汉中市行政事业单位国有资产管理委员会办公室四级主任科员公务员招录1人《行政职业能力测验》模拟试卷(答案详解版)
- 艺术培训校长述职报告
- 选择性必修一 期末综合测试(二)(解析版)2021-2022学年人教版(2019)高二数学选修一
- 《论语》学而篇-第一课件
- 《写美食有方法》课件
- 学校制度改进
- 各行业智能客服占比分析报告
- 年产30万吨高钛渣生产线技改扩建项目环评报告公示
- 心电监护考核标准
- (完整word版)申论写作格子纸模板
评论
0/150
提交评论