【SQL Server DBA】维护语句:删除并创建外键约束、获取建表语句
原文:【SQL Server DBA】维护语句:删除并创建外键约束、获取建表语句
1、删除外键约束,建立外键约束
先建立3个表:
-
/*
-
drop table tb
-
drop table tb_b
-
drop table tb_c
-
*/
-
-
--建立3个关联的表
-
create table tb(id int primary key ,vv varchar(10))
-
-
-
create table tb_b(
-
idd int primary key,
-
id int foreign key references tb(id)
-
)
-
-
create table tb_c(
-
iddd int primary key,
-
idd int foreign key references tb_b(idd)
-
)
-
go
可以生成删除外键的语句,需要复制出来,然后放到再执行:
-
;WITH FK --外键约束
-
AS
-
(
-
SELECT
-
SCH.name as foreign_schema_name, --外键schema名
-
FK.name as foreign_name, --外键名
-
FK.is_disabled , --是否禁用
-
-
FK.delete_referential_action_desc as delete_action,
-
FK.update_referential_action_desc as update_action,
-
-
FKC.constraint_column_id, --约束列的id
-
FKC.parent_object_id, --父对象的id
-
FKC.parent_column_id, --父对象列的id
-
-
-
FKC.referenced_object_id, --被引用的对象
-
FKC.referenced_column_id --被引用的对象中的列
-
FROM sys.foreign_keys FK
-
INNER JOIN sys.foreign_key_columns FKC
-
ON FK.object_id = FKC.constraint_object_id
-
INNER JOIN sys.schemas SCH
-
ON FK.schema_id = SCH.schema_id
-
),
-
-
TB --表和列
-
AS
-
(
-
SELECT
-
TB.object_id,
-
SCH.name as schema_name,
-
TB.name as table_name,
-
C.column_id as column_id,
-
C.name as column_name
-
FROM sys.tables TB WITH(NOLOCK)
-
INNER JOIN sys.columns C WITH(NOLOCK)
-
ON TB.object_id = C.object_id
-
INNER JOIN sys.schemas SCH WITH(NOLOCK)
-
ON TB.schema_id = SCH.schema_id
-
WHERE TB.is_ms_shipped = 0 -- 此条件表示仅查询不是由内部 SQL Server 组件创建对象
-
)
-
-
SELECT
-
'alter table ['+TBP.schema_name+'].['+TBP.table_name+
-
'] drop constraint ['+FK.foreign_name+'];' as '删除外键的语句,复制出来后运行'
-
FROM FK
-
INNER JOIN TB TBP
-
ON FK.parent_object_id = TBP.object_id
-
AND FK.parent_column_id = TBP.column_id
-
-
INNER JOIN TB TBR
-
ON FK.referenced_object_id = TBR.object_id
-
AND FK.referenced_column_id = TBR.column_id
-
/*
-
删除外键的语句,复制出来后运行
-
alter table [dbo].[tb_b] drop constraint [FK__tb_b__id__6754599E];
-
alter table [dbo].[tb_c] drop constraint [FK__tb_c__idd__6C190EBB];
-
*/
另外,删除主键后,插入数据,然后再建立外键:
-
;WITH FK --外键约束
-
AS
-
(
-
SELECT
-
SCH.name as foreign_schema_name, --外键schema名
-
FK.name as foreign_name, --外键名
-
FK.is_disabled , --是否禁用
-
-
FK.delete_referential_action_desc as delete_action,
-
FK.update_referential_action_desc as update_action,
-
-
FKC.constraint_column_id, --约束列的id
-
FKC.parent_object_id, --父对象的id
-
FKC.parent_column_id, --父对象列的id
-
-
-
FKC.referenced_object_id, --被引用的对象
-
FKC.referenced_column_id --被引用的对象中的列
-
FROM sys.foreign_keys FK
-
INNER JOIN sys.foreign_key_columns FKC
-
ON FK.object_id = FKC.constraint_object_id
-
INNER JOIN sys.schemas SCH
-
ON FK.schema_id = SCH.schema_id
-
),
-
-
TB --表和列
-
AS
-
(
-
SELECT
-
TB.object_id,
-
SCH.name as schema_name,
-
TB.name as table_name,
-
C.column_id as column_id,
-
C.name as column_name
-
FROM sys.tables TB WITH(NOLOCK)
-
INNER JOIN sys.columns C WITH(NOLOCK)
-
ON TB.object_id = C.object_id
-
INNER JOIN sys.schemas SCH WITH(NOLOCK)
-
ON TB.schema_id = SCH.schema_id
-
WHERE TB.is_ms_shipped = 0 -- 此条件表示仅查询不是由内部 SQL Server 组件创建对象
-
)
-
-
SELECT
-
'alter table ['+TBP.schema_name+'].['+TBP.table_name+
-
'] add constraint ['+FK.foreign_name+'] '+
-
' foreign key('+TBP.column_name+') references [' +
-
TBR.schema_name +'].['+ TBR.table_name +']('+TBR.column_name+')'
-
as '新建外键索引,复制然后在运行'
-
FROM FK
-
INNER JOIN TB TBP
-
ON FK.parent_object_id = TBP.object_id
-
AND FK.parent_column_id = TBP.column_id
-
-
INNER JOIN TB TBR
-
ON FK.referenced_object_id = TBR.object_id
-
AND FK.referenced_column_id = TBR.column_id
-
/*
-
新建外键索引,复制然后在运行
-
alter table [dbo].[tb_c] add constraint [FK__tb_c__idd__0A9D95DB] foreign key(idd) references [dbo].[tb_b](idd)
-
alter table [dbo].[tb_b] add constraint [FK__tb_b__id__05D8E0BE] foreign key(id) references [dbo].[tb](id)
-
*/
另外,还有一个问题:原来两个表之间是有外键的,删除了外键导入数据后,导入的数据记录条数和原表也一致,发现子表有记录不属于主表的,那么原来的外键是怎么建立的?
-
create table tb(id int primary key ,vv varchar(10))
-
-
insert into tb
-
values(1,'aa')
-
go
-
-
create table tb_b(
-
idd int primary key,
-
id int --foreign key references tb(id)
-
)
-
-
insert into tb_b
-
values(1,2) --id不在主表中
-
go
-
-
--新增外键约束,不会报错,with nocheck对于之前已经存在的数据,不会进行检测
-
ALTER TABLE [dbo].[tb_b] WITH noCHECK ADD FOREIGN KEY([id])
-
REFERENCES [dbo].[tb] ([id])
-
GO
-
-
-
--会报错 ,在建立上面的约束后,再次插入,就会报错了
-
insert into tb_b
-
values(2,2) --id不在主表中
2、如何根据表名查询出创建该表的代码
-
--当用以下代码创建一个表后,如何根据表名查询出创建该表的代码(也就是以下代码)?
-
CREATE TABLE [dbo].[a1](
-
[c2] [decimal](10, 2) NULL,
-
[c3] [decimal](10, 3) NULL CONSTRAINT [DF_a1_c3] DEFAULT ((0)),
-
[re] [bigint] IDENTITY(1,1) NOT NULL,
-
CONSTRAINT [PK_a1] PRIMARY KEY CLUSTERED
-
(
-
[re] 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
-
declare @sql varchar(8000),@tablename varchar(100)
-
set @tablename ='a1'--这里输入表名
-
set @sql = 'create table ['+@tablename+']
-
(
-
'
-
select @sql = @sql + b.name + ' '+
-
c.name+
-
case when c.collation_name is not null then '('+
-
case when b.max_length <>-1 then convert(varchar(100),b.max_length)
-
else 'MAX'
-
end +') '
-
else ''
-
end +
-
case when b.is_identity = 1 then ' identity('+convert(varchar(100),IDENT_SEED(@tablename))+','+convert(varchar(100),IDENT_INCR(@tablename))+')' else '' end +
-
case when d.definition is not null then ' default('+d.definition +')' else '' end +
-
case when b.is_nullable = 0 then ' not null' else ' null' end +
-
-
',
-
'
-
from sys.objects a join sys.columns b
-
on a.object_id = b.object_id
-
join sys.types c
-
on b.system_type_id = c.system_type_id and b.user_type_id = c.user_type_id
-
left join sys.default_constraints d
-
on b.default_object_id = d.object_id
-
where a.name=@tablename
-
order by b.column_id
-
if exists(select * from sys.indexes where object_id =object_id(@tablename) and is_primary_key =1 )
-
begin
-
select @sql = @sql + 'CONSTRAINT ['+name+'] PRIMARY KEY '+type_desc+'
-
(
-
' from sys.indexes where object_id =object_id(@tablename) and is_primary_key =1
-
select @sql = @sql + b.name + case when a.is_descending_key =1 then ' DESC' else ' ASC' end +',
-
' from sys.index_columns a join sys.columns b
-
on a.object_id= b.object_id and a.column_id = b.column_id
-
where a.object_id =object_id(@tablename)
-
select @sql = left(@sql,len(@sql)-3)+'
-
)'
-
select @sql = @sql+'
-
) ON [PRIMARY] '
-
end
-
else
-
begin
-
select @sql = left(@sql,len(@sql)-1)+'
-
) ON [PRIMARY] '
-
end
-
print @sql
-
-
/*
-
create table [a1]
-
(
-
c2 decimal null,
-
c3 decimal default(((0))) null,
-
re bigint identity(1,1) not null,
-
CONSTRAINT [PK_a1] PRIMARY KEY CLUSTERED
-
(
-
re ASC
-
)
-
) ON [PRIMARY]
-
-
*/
【SQL Server DBA】维护语句:删除并创建外键约束、获取建表语句的更多相关文章
- sql server drop talbe 自动删除关联的外键 ,权限体系(二)
alter table dbo.Sys_PowerTeamForUser add constraint FK_Sys_User_Sys_PowerTeamForUser foreign key (Sy ...
- sql server drop talbe 自动删除关联的外键 ,权限体系(一)
if object_id('Proc_DropTableWithFK') is not null begin drop proc dbo.Proc_DropTableWithFK end GO ) a ...
- sql操作数据库(3)-->外键约束、数据库表之间的关系、三大范式、多表查询、事务
外键约束 在新表中添加外键约束语法: constraint 外键约束名称 foreign key(外键的字段名称) references 主表表名(主键字段名) 在已有表中添加外键约束:alter t ...
- SQL-46 在audit表上创建外键约束,其emp_no对应employees_test表的主键id。
题目描述 在audit表上创建外键约束,其emp_no对应employees_test表的主键id.CREATE TABLE employees_test(ID INT PRIMARY KEY NOT ...
- C# 如何物理删除有主外键约束的记录?存储过程实现
十年河东,十年河西,莫欺少年穷 本篇主旨是如何物理删除有主外键约束的记录!那么,我们从主外键走起! 下面新建三张有主外键约束的表,分别为:系/学院表,专业班表,学生表,如下: CREATE TABLE ...
- SQL语句删除和添加外键、主键的方法
--删除外键 语法:alter table 表名 drop constraint 外键约束名 如: alter table Stu_PkFk_Sc drop constraint FK_s alter ...
- 【转】SQL语句删除和添加外键、主键
--删除外键 语法:alter table 表名 drop constraint 外键约束名 如: alter table Stu_PkFk_Sc drop constraint FK_s alter ...
- MySQL数据库 : 查询语句,连接查询及外键约束
查询指定字段 select 字段1,字段2 from 表名; 消除重复行(重复指的是结果集中的所有完全重复行) select distinct 字段1,字段2.. ...
- MySQL创建外键约束的报错Error : Can't create table '#sql-534_185' (errno: 150)
总得来说是因为两个表的字段类型不一致,例如: 两个字段的类型或大小不严格匹配,一个为tinyint,另一个为char:或一个为int(10)另一个为int(9)也是不行的,即使都为int(10),但一 ...
随机推荐
- ArcPy地理处理工具案例教程—批量添加栅格数据
ArcPy地理处理工具案例教程-批量添加栅格数据 商务合作,科技咨询,版权转让:向日葵,135-4855__4328,xiexiaokui#qq.com 关键字: Arcpy,python,地理处理工 ...
- Centos7迁移fastdfs文件系统
系统从一个地方迁移到另一个地方,数据保持不变,但是ip地址和网络情况不一样了,最困难的是要迁移的那个地方还么有互联网,这TM就坑了,所以想到将FastDFS存储的目录整体拷贝过去,这个方法简单粗暴,这 ...
- 如何贡献补丁到uboot社区?
答: 首次贡献分为两步: 1. 首先需要订阅一下,地址在此https://lists.denx.de/listinfo/u-boot,使邮箱地址对应有一个成员名称,才能向uboot社区发送补丁,否则会 ...
- Kotlin集合——Map集合
Kotlin集合——Map集合 转 https://www.jianshu.com/p/da5cc9072f1e Kotlin的Map集合用于保存key-value对,其也被分为可变的和不可变的. 一 ...
- shell编程系列7--shell中常用的工具find、locate、which、whereis
shell编程系列7--shell中常用的工具find.locate.which.whereis .文件查找之find命令 语法格式:find [路径] [选项] [操作] 选项 -name 根据文件 ...
- Spring Cloud Eureka 服务发现 4.2
在微服务架构中,服务发现可以说是最为核心和基础的模块,该模块主要用于实现各个微服务实例的自动化注册与发现.在Spring Cloud的子项目中,Spring Cloud Netflix提供了Eur ...
- Nginx 504响应超时
1.问题分析 nginx访问出现504 Gateway Time-out,一般是由于程序执行时间过长导致响应超时,例如程序需要执行90秒,而nginx最大响应等待时间为30秒,这样就会出现超时. ...
- 【原生JS插件】LoadingBar页面顶部加载进度条
先展示一下已经实现的效果: 预览地址:http://dtdxrk.github.io/js-plug/LoadingBar/index.html 看到手机上的浏览器内置了页面的加载进度条,想用在pc上 ...
- .net core 使用SignalR实现实时通信
这几天在研究SignalR,网上大部分的例子都是聊天室,我的需求是把服务端的信息发送给前端展示.并且需要实现单个用户推送. 用户登录我用的是ClaimsIdentity,这里就不多解释,如果不是很了解 ...
- 【VS开发】开发最小化到托盘的功能
在VC++中,想实现最小化MFC程序的时候,最小化到系统托盘,需要调用NOTIFYICONDATA类 下面我们就来讲解一下如何简单实现一个系统托盘我们以对话框程序为列 第一步:在Dlg类中//定义一个 ...
