这个案例是前几天同事遇到的一个案例,在存储过程中“删除”了一个临时表,然后重新创建这个临时表时遇到“There is already an object named 'xxxx' in the database."这样的错误。下面简单演示一下这个案例(不用存储过程,而是直接用简单的SQL语句重现)

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

 

 

IF  EXISTS (SELECT 1 FROM tempdb..sysobjects WHERE name='#tmp_test')

BEGIN

    DROP TABLE #temp_test;

END

GO

 

 

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

同事很是不解,问我为什么在存储过程里面“删除”了这个临时表居然不能重建呢? 其实这里面并没有什么玄机,而是仅仅犯了逻辑错误。上面这个SQL语句,其实永远也不会删除这个临时表。

原因很简单,临时表#temp_test在tempdb..sysobjects里面存储的名字为“#temp_test_________________________________________________________________0000000000EE”,所以上述脚本犯了一个逻辑错误: 开发人员以为临时表被删除了,其实实质上永远不会执行DROP TABLE #temp_test这句SQL。存储在tempdb的sysobjects表中的临时表,其全名由 CREATE TABLE 语句中指定的表名和系统生成的数字后缀组成。为了允许追加后缀,为本地临时表指定的table_name不能超过 116 个字符(关于为什么是116字符,也是有原因的,此处不展开)

Both regular and delimited identifiers must contain from 1 through 128 characters. For local temporary tables, the identifier can have a maximum of 116 characters.

那么为什么数据库自动会给临时表这样命名呢? 因为本地临时表是对当前连接(当前会话)可见的。但是任意会话都可以创建同样名字的临时表,那么临时表的元数据在数据库内部存储的时候,使用同样的名字就不能定位和区别,所以设计上,为了区别不同会话下同样命名的临时表,在tempdb..sysobjects里面临时表会保存的是数据库自动生成的名字(#临时表名+下划线+12位的十六进制字符),如下测试所示,在另外一个会话中,我们创建一个同样名字的临时表,然后我们去数据库检查,你会发现tempdb..sysobjects中有两个对象,他们名字的后缀是不一样的。

SELECT @@SPID;

GO

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

那么要如何定位、查找当前会话是否创建了本地临时表对象呢? 其实使用OBJECT_ID函数就OK了:

IF OBJECT_ID('tempdb..#temp_test' ) IS NOT NULL DROP TABLE #temp_test;

那么存储过程中使用上述脚本删除本地临时表,然后重建同样名字的临时表就可以了吗? 事实告诉我们,虽然上面SQL可以找到当前会话创建的本地临时表删除,但是如果是在存储过程里面,使用这种方式,创建本地临时表,然后删除、创建,依然会遇到这个错误,如下测试所示:

CREATE PROCEDURE PRC_TEST

AS

BEGIN 

 

    IF OBJECT_ID('tempdb..#temp_test' ) IS NOT NULL DROP TABLE #temp_test;

    CREATE TABLE #temp_test( id INT, name VARCHAR(32));

 

    INSERT INTO #temp_test VALUES(10, 'jimmy');

 

    IF OBJECT_ID('tempdb..#temp_test' ) IS NOT NULL DROP TABLE #temp_test;

 

    CREATE TABLE #temp_test( id INT, name VARCHAR(32));

 

    INSERT INTO #temp_test VALUES(100, 'kerry');

 

 

    DROP TABLE #temp_test;

END

GO

其实在一个批处理里面执行下面SQL语句也会报错,但是改写一下SQL,在删除后面加上一个GO语句,那么这样是不会报错的(但是你单个SQL,一条一条执行是不会报错的).

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

INSERT INTO #temp_test VALUES(10, 'jimmy');

IF OBJECT_ID('tempdb..#temp_test' ) IS NOT NULL DROP TABLE #temp_test;

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

INSERT INTO #temp_test VALUES(100, 'kerry');

DROP TABLE #temp_test;

至于原因是什么呢?网上有种分析是因为解析错误(parse error),这里我也倾向于这种说法,因为测试过程中,发现其实上面SQL语句报错,但是实质上,本地临时表已经在tempdb被删除了。加上一个GO这种改写方法,其实使用两个批处理,下面这样的SQL是不会报错的。

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

INSERT INTO #temp_test VALUES(10, 'jimmy');

IF OBJECT_ID('tempdb..#temp_test' ) IS NOT NULL DROP TABLE #temp_test;

  GO

CREATE TABLE #temp_test( id INT, name VARCHAR(32));

INSERT INTO #temp_test VALUES(100, 'kerry');

DROP TABLE #temp_test;

参考资料:

https://blog.sqlauthority.com/2012/05/01/sql-server-maximum-allowable-length-of-characters-for-temp-objects-is-116-guest-post-by-balmukund-lakhani/

https://blogs.msdn.microsoft.com/sqlserverfaq/2012/03/15/an-interesting-find-about-temp-tables-in-sql-server/

There is already an object named '#xxxx' in the database.的更多相关文章

  1. mybatis There is no getter for property named 'xxxx

    mybatis There is no getter for property named 'xxxx 360反馈意见截图16230322799670.png http://blog.sina.com ...

  2. python3命令行ImportError: No module named 'xxxx'的问题

    主要原因:启动脚本不在当前目录下,无法找到上一层 在pycharm写好的脚本程序,在命令行无法运行,报错 Traceback (most recent call last): File "t ...

  3. python中类的输出或类的实例输出为何是<__main__类名 object at xxxx>这种形式?

    原因: __str__()这个特殊方法将对象转换为字符串的结果 效果图: 代码: # 定义一个Person类 class Person(object): """人类&qu ...

  4. 问题: 刚安装的PyCharm执行代码报“ModuleNotFoundError: No module named XXXX”错

    老猿刚安装好PyCharm后,直接新建了一个工程文件并导入了一个已有的爬虫程序文件,该文件原来在Python解释器下能执行,但在PyCharm下执行时报错: F:\学习\python\SRC\proj ...

  5. #!/usr/bin/python3的作用 解决vscode ImportError: No module named xxxx

    在 Python 脚本的第一行经常见到这样的注释: #!/usr/bin/env python3 或者 #!/usr/bin/python3 含义 在脚本中, 第一行以 #! 开头的代码, 在计算机行 ...

  6. 命令行运行python项目文件,报错:ModuleNotFoundError: No module named 'xxxx' 解决办法

    在pycharm中写好了自动化测试脚本,并能在pycharm中正常运行,由于要考虑到无人值守时能自动执行,执行时就需要脱离pycharm,直接能用命令执行.但是直接用命令执行用例文件:python3 ...

  7. 【Entity Framework】disable automatic migration, 执行update-migration仍然会显示有automatic migration

    本文涉及的相关问题,如果你的问题或需求有与下面所述相似之处,请阅读本文 [Entity Framework] disable automatic migration, 执行update-migrati ...

  8. MVC5 网站开发之七 用户功能 1、角色的后台管理

    角色是网站中都有的一个功能,用来区分用户的类型.划分用户的权限,这次实现角色列表浏览.角色添加.角色修改和角色删除. 目录 奔跑吧,代码小哥! MVC5网站开发之一 总体概述 MVC5 网站开发之二 ...

  9. Constraint2:constraint

    一,Constraint 是表定义的一部分,用于实现数据完整性. Data Integrity 由三种类型的constraint实现: Entity Integrity:数据是唯一的.约束: prim ...

随机推荐

  1. python之循序渐进学习装饰器

    python装饰器的定义:在代码运行期间在不改变原函数定义的基础上,动态给该函数增加功能的方式称之为装饰器(Decorator) 装饰器的优点和用途: 1. 抽离出大量函数中与函数功能本身无关的的雷同 ...

  2. idea配置tomcat运行按钮置灰,下拉没有自定义的tomcat选项

    一.问题 下拉没有自定义tomcat的选项 run按钮置灰,点不了 二.解决 添加自己的tomcat时,一定要点加号,不要用那个默认的.

  3. MSSQL 更改表结构

    更改表结构: alter TABLE 表1 ALTER COLUMN 列名1 NCHAR(40)

  4. 'module' object has no attribute 'Thread'解决方法及模块加载顺序

    源码片段: class myThread(threading.Thread): def __init__(self, threadID, name, counter): threading.Threa ...

  5. python安装第三方库报错visual c++ 14.0 is required

    使用python安装第三方库时报错如下: error: Microsoft Visual C++ 14.0 is required. Get it with “Microsoft Visual C++ ...

  6. 【python3基础】相对路径,‘/’,‘./’,‘../’

    python3相对路径 “/” 前有没有 “.” ,有几个“.”,意思完全不一样. “/”:表示根目录,在windows系统下表示某个盘的根目录,如“E:\”: “./”:表示当前目录:(表示当前目录 ...

  7. 小乌龟git

    一.概念 分布式版本控制系统.诞生于Linux社区,有兴趣可以去了解下git的前世今生. 与集中式版本控制有明显区别.集中式:集中管理的服务器,保存所有的修订版本,协同工作的人通过客户端连接服务器,取 ...

  8. logisim元件清单

  9. Android Studio开发环境配置以及相关说明

    版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 这里简单记录下在开发的时候使用的Android Studio开发环境版本以及相关注意事项. 一般来讲,每隔一段时间就要检查下Andr ...

  10. UmengShareDemo【友盟分享SDK集成,基于V6.9.3版本】

    版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 这里简单记录下友盟分享SDK集成的步骤. 如果想要使用自定义分享对话框实现,请参考<ShareDialogDemo[分享对话框 ...