在SQL Server中,实际上外键值可不可以为空(NULL),和外键关系是不是强制约束无关。

我们先在SQL Server数据库中建立两张表People和Car,一个People可以有多个Car,所以这两张表是一对多关系。

建立表

People建表语句:

CREATE TABLE [dbo].[People](
[ID] [int] NOT NULL,
[Name] [nvarchar](50) NULL,
[Age] [int] NULL,
[Sex] [nvarchar](50) NULL,
CONSTRAINT [PK_People] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO

其中列ID是People表的主键

Car建表语句:

CREATE TABLE [dbo].[Car](
[ID] [int] NOT NULL,
[Brand] [nvarchar](50) NULL,
[PeopleID] [int] NULL,
CONSTRAINT [PK_Car] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO ALTER TABLE [dbo].[Car] WITH CHECK ADD CONSTRAINT [FK_Car_People] FOREIGN KEY([PeopleID])
REFERENCES [dbo].[People] ([ID])
GO ALTER TABLE [dbo].[Car] CHECK CONSTRAINT [FK_Car_People]
GO

其中列ID是Car表的主键,此外列PeopleID是Car表的外键,其关联People表的主键ID。我们还为People表和Car表之间的外键关系[FK_Car_People]设置了强制约束(CHECK)。

注意上面外键约束[FK_Car_People]中CHECK关键字代表是强制约束,而如果使用NOCHECK关键字代表是非强制约束。

外键关系语法扩展

建立外键关系[FK_Car_People]为强制约束(CHECK)的SQL语句如下:

--下面CHECK关键字表示建立外键关系时就开启强制约束
ALTER TABLE [dbo].[Car] WITH CHECK ADD CONSTRAINT [FK_Car_People] FOREIGN KEY([PeopleID])
REFERENCES [dbo].[People] ([ID])
GO --注意上面建立外键关系[FK_Car_People]后,下面这句ALTER TABLE语句也不能少
ALTER TABLE [dbo].[Car] CHECK CONSTRAINT [FK_Car_People]
GO

建立外键关系[FK_Car_People]为非强制约束(NOCHECK)的SQL语句如下:

--下面NOCHECK关键字表示建立外键关系时不开启强制约束,也就是非强制约束
ALTER TABLE [dbo].[Car] WITH NOCHECK ADD CONSTRAINT [FK_Car_People] FOREIGN KEY([PeopleID])
REFERENCES [dbo].[People] ([ID])
GO --注意上面建立外键关系[FK_Car_People]后,下面这句ALTER TABLE语句也不能少
ALTER TABLE [dbo].[Car] NOCHECK CONSTRAINT [FK_Car_People]
GO

上面建立好外键关系[FK_Car_People]后,可以通过下面的语句再将外键关系[FK_Car_People]改为强制(CHECK)或非强制(NOCHECK)约束,其实就是删除外键关系[FK_Car_People]后再重建:

更改外键关系[FK_Car_People]为强制约束(CHECK)的SQL语句如下:

--删除外键关系[FK_Car_People]
ALTER TABLE [dbo].[Car] DROP CONSTRAINT [FK_Car_People] --下面CHECK关键字表示建立外键关系时就开启强制约束
ALTER TABLE [dbo].[Car] WITH CHECK ADD CONSTRAINT [FK_Car_People] FOREIGN KEY([PeopleID])
REFERENCES [dbo].[People] ([ID])
GO --注意上面建立外键关系[FK_Car_People]后,下面这句ALTER TABLE语句也不能少
ALTER TABLE [dbo].[Car] CHECK CONSTRAINT [FK_Car_People]
GO

更改外键关系[FK_Car_People]为非强制约束(NOCHECK)的SQL语句如下:

--删除外键关系[FK_Car_People]
ALTER TABLE [dbo].[Car] DROP CONSTRAINT [FK_Car_People] --下面NOCHECK关键字表示建立外键关系时不开启强制约束,也就是非强制约束
ALTER TABLE [dbo].[Car] WITH NOCHECK ADD CONSTRAINT [FK_Car_People] FOREIGN KEY([PeopleID])
REFERENCES [dbo].[People] ([ID])
GO --注意上面建立外键关系[FK_Car_People]后,下面这句ALTER TABLE语句也不能少
ALTER TABLE [dbo].[Car] NOCHECK CONSTRAINT [FK_Car_People]
GO

在本例中,外键关系[FK_Car_People]应该是强制约束(CHECK)的,所以下面我们来为People表和Car表插入数据。

插入数据

下面我们为People表插入两条数据:

INSERT INTO [dbo].[People](ID,Name,Age,Sex)
VALUES
(1,N'张三',25,N'男'),
(2,N'李四',26,N'女');

查询结果如下:

然后我们为Car表插入五条数据:

INSERT INTO [dbo].[Car](ID,Brand,PeopleID)
VALUES
(1,N'奔驰',1),
(2,N'宝马',1),
(3,N'大众',2),
(4,N'别克',NULL),
(5,N'丰田',NULL);

注意我们在最后两条数据"别克"和"丰田"中,插入了空值(NULL)到Car表的外键列PeopleID。但是语句并没有报错,五条数据都被成功插入了,查询结果如下:

所以Car表的外键列PeopleID能不能为空(NULL),和其外键关系[FK_Car_People]是不是强制约束(CHECK)无关,它只和Car表允不允许外键列PeopleID为空(NULL)相关。

现在我们把Car表的外键列PeopleID改为不允许为空(NULL),然后再次插入前面的五条数据:

DELETE FROM [dbo].[Car]

ALTER TABLE [dbo].[Car]
ALTER COLUMN [PeopleID] INT NOT NULL INSERT INTO [dbo].[Car](ID,Brand,PeopleID)
VALUES
(1,N'奔驰',1),
(2,N'宝马',1),
(3,N'大众',2),
(4,N'别克',NULL),
(5,N'丰田',NULL);

这时插入语句就会报错了,因为此时Car表的外键列PeopleID不能为空(NULL):

所以SQL Server中的外键关系强制约束,约束的实际上是外键的非空值,而外键的空值(NULL)并不受外键强制约束。

SQL Server外键关系是强制约束,外键值也可以是空(NULL)的更多相关文章

  1. 在SQL Server中实现关系模型的阶梯到级别3的t -SQL DML

    在SQL Server中实现关系模型的阶梯到级别3的t -SQL DML 格雷戈里·拉森(Gregory Larsen),2017/08/02(第一次出版:2011 /11/09) 原文链接:http ...

  2. 在SQL Server中实现关系模型

    使用SQL Server的Transact-SQL(T-SQL)方言,此楼梯将为您提供如何使用SQL Server表中的数据的基本了解. DML是数据操作语言,是处理数据的语言的一个方面.它包括SEL ...

  3. 到T-SQL DML 三级的阶梯:在SQL server中实现关系模型

    作者: Gregory Larsen, 2017/08/02 (第一次出版: 2011/11/09) 翻译:谢雪妮,许雅莉,赖慧芳,刘琼滨 译文: 系列 该文章是阶梯系列的一部分:T-SQL DML的 ...

  4. SQL Server建库-建表-建约束

    ----------------------------------------SQL Server建库-建表-建约束创建School数据库------------------------------ ...

  5. SQL Server 按某一字段分组 取 最大 (小)值所在行的数据

    SQL Server 按某一字段分组 取 最大 (小)值所在行的数据 -- 按某一字段分组 取 最大 (小)值所在行的数据 -- (爱新觉罗.毓华(十八年风雨,守得冰山雪莲花开) 2007-10-23 ...

  6. SQL SERVER将某一列字段中的某个值替换为其他的值 分类: MSSQL 2014-11-05 13:11 67人阅读 评论(0) 收藏

    SQL SERVER将某一列字段中的某个值替换为其他的值 UPDATE 表名 SET 列名 = REPLACE(列名 ,'贷','袋') SQL SERVER"函数 replace 的参数 ...

  7. SQL Server 中几种常见的约束关系

    1.创建唯一约束 当表中已创建主键,但又要保证其他数据列的值唯一时,可以使用唯一约束,并且唯一约束允许NULL值(只有一个) (1)展开指定的数据库: (2)右击要创建唯一约束的表,在弹出的快捷菜单中 ...

  8. sql server drop talbe 自动删除关联的外键 ,权限体系(一)

    if object_id('Proc_DropTableWithFK') is not null begin drop proc dbo.Proc_DropTableWithFK end GO ) a ...

  9. 【2017-03-09】SQL Server 数据库基础、四种约束

    一.数据库和内存的区别 数据库:一些存储在硬盘上的数据文件 内存:计算机临时存储的一些数据 二.常用数据库 .Net - SQL Server PHP - MySql Java - Oreacl 三. ...

随机推荐

  1. Django学习之十四:Django ORM继承关系

    目录 Django ORM继承关系 1. SINGLE_TABLE(django好像不支持) 2. TABLE_PER_CLASS 3. JOINED 4. 代理继承 Django ORM继承关系 参 ...

  2. 并发编程(六)--进程/线程池、协程、gevent第三方库

    一.进程/线程池 1.进程池 (1)什么是进程池 如果需要创建的子进程数量不大,可以直接利用multiprocess中的Process来创建.但是当需要创建上百个或上千个,手动创建就较为繁琐,这时就可 ...

  3. Nginx+Mysql调优

    使用nginx实现反向代理作用,具备负载均衡的功能.     接受客户端的请求 | nginx(宿主机) | |-------------------| web1 web2 (客户机)   原理: 与 ...

  4. 如何检查linux服务器是否被入侵

    被入侵服务器的症状 当服务器被没有经验攻击者或者自动攻击程序入侵了的话,他们往往会消耗 100% 的资源.他们可能消耗 CPU 资源来进行数字货币的采矿或者发送垃圾邮件,也可能消耗带宽来发动 DoS ...

  5. HDU6583:Typewriter(dp+后缀自动机)

    传送门 题意: 给出\(p,q\),现在要你生成一个字符串\(s\). 你可以进行两种操作:一种是花费\(p\)的代价随意在后面添加一个字符,另一种是花费\(q\)的代价可以随意赋值前面的一个子串. ...

  6. python的continue和pass的区别

    a = 'pythyon' i = 2 for element in a: if element == 'y': pass i = 3 else: print(element+str(i)) 结果: ...

  7. USACO Chocolate Giving

    洛谷 P2984 [USACO10FEB]给巧克力Chocolate Giving 洛谷传送门 JDOJ 2680: USACO 2010 Feb Silver 2.Chocolate Giving ...

  8. [Noip2018]填数游戏

    传送门 Description 耳熟能详,就不多说了 Solution 对于一个不会推式子的蒟蒻,如何在考场优雅地通过此题 手玩样例,发现对于 \(n=1\) , \(ans=2^m\) .对于 \( ...

  9. 【IntelliJ IDEA学习之八】版本控制之SVN

    版本:IntelliJIDEA2018.1.4 一.SVN1.安装SVN客户端,command line client tools默认是不安装的,这里要勾选上(不用重新卸载安装,只找到安装程序,选择  ...

  10. 【2019年06月28日】A股最便宜的股票

    查看更多A股最便宜的股票:androidinvest.com/CNValueTop/ 经典价值三因子选股: 市盈率PE.市净率PB 和 股息分红率,按照 1:1:1的权重,选择前10大最便宜的股票. ...