SQL Server现有表上自增属性增删原理研究
项目需求:线上有一张表,数据类型为int类型,现在由于项目变更,需要这一列添加自增属性,而且,为了保证能尽快完成,希望使用脚本来实现,而不是在表设计中通过GUI窗口来实现。
问题来了:SQL Server有类似于alter table的语法来直接修改表的列为自增列的吗?答案是:没有!那么,表设计中是如何实现的呢?
创建一张测试表t1,然后使用SQL Server Profile来看看内部是怎么实现的。
一、对现有的列添加自增属性
步骤一:创建测试表t1
--表若存在,就删除
if(object_id('t1') is not null)
begin
drop table t1 ;
end ; --创建测试表
create table t1(id int, c1 char(10), c2 char(10)) ; --插入测试数据
insert into t1(id, c1, c2) values(1,'aaaaaaaaaa','bbb'),(10,'aaaaaaaaaa','bbb'),(100,'aaaaaaaaaa','bbb'),(1000,'aaaaaaaaaa','bbb');
步骤二:开启SQL Server Profile(略)
步骤三:打开表设计的GUI界面,修改为ID列为自增列(如下图红色框内所示),crtl+S保存一下
步骤四:保存完成后,停止SQL Server Profile跟踪,查看SQL Server内部是怎么实现的
总共是七步,详细步骤描述如下
--1、创建与原表表结构一致的临时表,并且在列上添加了自增属性
CREATE TABLE dbo.Tmp_t1
(
id int NOT NULL IDENTITY (10, 1),
c1 char(10) NULL,
c2 char(10) NULL
) ON [PRIMARY] --2、把新增临时表的锁升级为表锁
ALTER TABLE dbo.Tmp_t1 SET (LOCK_ESCALATION = TABLE) --3、设置新增临时表的自增列为可插入状态
SET IDENTITY_INSERT dbo.Tmp_t1 ON --4、把原表中的数据插入到临时表里
IF EXISTS(SELECT * FROM dbo.t1)
EXEC('INSERT INTO dbo.Tmp_t1 (id, c1, c2)
SELECT id, c1, c2 FROM dbo.t1 WITH (HOLDLOCK TABLOCKX)') --5、设置新增临时表的自增列为不可插入状态
SET IDENTITY_INSERT dbo.Tmp_t1 OFF --6、删除原表
DROP TABLE dbo.t1 --7、把临时表的表名修改为跟原表一致
EXECUTE sp_rename N'dbo.Tmp_t1', N't1', 'OBJECT'
可见,SQL Server内部也是通过使用临时表作为中转来实现把列修改为自增列的,所以,如果真的需要用脚本而非GUI来实现修改为自增列,可以参考以上的7个步骤。
注意:在设计中是修改了标识种子为10,所以在创建临时表Tmp_t1的时候出现了IDENTITY(10,1),如果没有修改标识种子,默认的是IDENTITY(1,1),可以在修改完成后使用以下语句进行修改
--修改自增列的标识种子
DBCC CHECKIDENT('t1', reseed, 100) ; --查看自增列的当前值
SELECT IDENT_CURRENT('t1')
二、对现有的列删除自增属性
步骤一:开启SQL Server Profile(略)
步骤二、打开表设计的GUI界面,修改为ID列为非自增列(如下图红色框内所示),crtl+S保存一下
步骤三、保存完成后,停止SQL Server Profile跟踪,查看SQL Server内部是怎么实现的
总共是6步,详细描述如下
--创建临时表dbo.Tmp_t1
CREATE TABLE dbo.Tmp_t1
(
id int NOT NULL,
c1 varchar(20) NULL,
c2 varchar(20) NULL
) ON [PRIMARY] --锁定临时表,锁级别为表锁
ALTER TABLE dbo.Tmp_t1 SET (LOCK_ESCALATION = TABLE) --把原来的表的数据插入到临时表dbo.Tmp_t1
IF EXISTS(SELECT * FROM dbo.t1)
EXEC('INSERT INTO dbo.Tmp_t1 (id, c1, c2)
SELECT id, c1, c2 FROM dbo.t1 WITH (HOLDLOCK TABLOCKX)') --删除原表
DROP TABLE dbo.t1 --将临时表改名为原表
EXECUTE sp_rename N'dbo.Tmp_t1', N't1', 'OBJECT' --添加索引
ALTER TABLE dbo.t1 ADD CONSTRAINT
PK__t1__3213E83F7F60ED59 PRIMARY KEY CLUSTERED
(
id
) WITH( STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
三、总结
从以上的跟踪结果可以看出,在SQL Server对于现有表的列进行添加或者删除自增属性,都是通过临时表作为中转表来实现的
以上,如有错谬,请不吝指正,万分感谢~~
SQL Server现有表上自增属性增删原理研究的更多相关文章
- 向SQL Server 现有表中添加新列并添加描述.
注: sql server 2005 及以上支持. 版本估计是不支持(工作环境2005,2008). 工作需要, 需要向SQL Server 现有表中添加新列并添加描述. 从而有个如下存储过程. (先 ...
- sql server 关于表中只增标识问题 C# 实现自动化打开和关闭可执行文件(或 关闭停止与系统交互的可执行文件) ajaxfileupload插件上传图片功能,用MVC和aspx做后台各写了一个案例 将小写阿拉伯数字转换成大写的汉字, C# WinForm 中英文实现, 国际化实现的简单方法 ASP.NET Core 2 学习笔记(六)ASP.NET Core 2 学习笔记(三)
sql server 关于表中只增标识问题 由于我们系统时间用的过长,数据量大,设计是采用自增ID 我们插入数据的时候把ID也写进去,我们可以采用 关闭和开启自增标识 没有关闭的时候 ,提示一下错 ...
- sql server 关于表中只增标识问题
由于我们系统时间用的过长,数据量大,设计是采用自增ID 我们插入数据的时候把ID也写进去,我们可以采用 关闭和开启自增标识 没有关闭的时候 ,提示一下错误,不能修改 set identity_inse ...
- SQL Server数据库表重置自增主键号(通常是指ID)
执行 DBCC CHECKIDENT ('table_name', NORESEED) 以确定列中的当前最大值 然后使用 DBCC CHECKIDENT ('table_name', RESEED,n ...
- [SQL]SQL Server数据表的基础知识与增查删改
SQL Server数据表的基础知识与增查删改 由张晨辉(学生) 于19天 前发表 | 阅读94次 一.常用数据类型 .整型:bigint.int.smallint.tinyint .小数:decim ...
- 干货 | RDS For SQL Server单库上云
数据库作为核心数据的重要存储,很多时候都会面临数据迁移的需求,例如:业务从本地迁移上云.数据中心故障需要切换至灾备中心.混合云或多云部署下的数据同步.流量突增导致数据库性能瓶颈需要拆分-- 本文将会一 ...
- SQL SERVER 数据库表同步复制 笔记
SQL SERVER 数据库表同步复制 笔记 同步复制可运行在不同版本的SQL Server服务之间 环境模拟需要两台数据库192.168.1.1(发布),192.168.1.10(订阅) 1.在发布 ...
- 在SQL Server中对视图进行增删改
原文:在SQL Server中对视图进行增删改 Lesktop开源IM发布以后,有一些网友问及如何在嵌入IM后与自己网站的用户系统整合(即如何让嵌入的IM直接使用网站原有的用户数据库,而不需要将已有的 ...
- Sql server 系统表
sql server系统表详细说明 SQL Server 用户库中系统表说明 名称 说明 备注 syscolumns 每个表和视图中的每列在表中占一行,存储过程中的每个参数在表中也占一行. sys ...
随机推荐
- UESTC - 878
状态的枚举还需多多练习啊 #include<iostream> #include<algorithm> #include<cstdio> #include<c ...
- IDEA查看第三方jar包的源代码时出现Decompiled.class file, bytecode version:52.0 (Java 8)的解决方案
IDEA中使用Ctrl+左键查看第三方jar包的源代码时,出现Decompiled.class file, bytecode version:52.0 (Java 8),说明IDEA没找到该类的.ja ...
- mysql大数据表删除操作锁表,导致其他线程等待锁超时(Lock wait timeout exceeded; try restarting transaction;)
背景: 1.有一个定时任务,每10分钟入一批统计数据: 2.另一个定时任务,每天定时清理7天前数据,此定时任务每天01:18:00执行: 现象: 每天01:20:00的统计数据入库失败,异常信息如下, ...
- nginx 服务器配置文件指令
localtion 配置 语法结构: location [ = ~ ~* ^~ ] uri{ ... } uri 变量是带匹配的请求字符, 可以是不含正则表达的字符串, ...
- 如何在vue && webpack 项目中的单文件组件中引入css
引入方式很简单,就是在script下使用require()即可. 因为import 是import...from 的形式,所以是不需要的. <script> import {mapStat ...
- 初探angular
前言 angular4.0目前已经发布了,angular是mvw框架,所以对其有一个简单的了解还是很有必要的. 目前angular有中文官网,且文档介绍也都是4.x的,但是为了了解其发展过程,我们先了 ...
- 使用jxl读取excel内容,并转换成Json,用于Datagrid
一.上传excel文件,得到InputStream,由InputStream得到Jxl中的Workbook,取出内容,存到二维数组中. 1.使用 Jquery Uploadify 插件(http:// ...
- memcached分布式部署
memcache和memcached两者使用起来几乎一模一样. $mem = new Memcache; $mem->addServer($memcachehost, '11211'); $me ...
- React.js 小书 Lesson15 - 实战分析:评论功能(二)
作者:胡子大哈 原文链接:http://huziketang.com/books/react/lesson15 转载请注明出处,保留原文链接和作者信息. 上一节我们构建了基本的代码框架,现在开始完善其 ...
- Python快速入门_1
注释 # 用#号字符开头注释单行 """ 三个引号可以注释多行 三个引号可以注释多行 三个引号可以注释多行 """ 原始数据类型和运算符 ( ...