这个案例是前几天同事遇到的一个案例,在存储过程中“删除”了一个临时表,然后重新创建这个临时表时遇到“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

    从2015开始国内就开始慢慢接触Python了,从16年开始Python就已经在国内的热度更高了,目前也可以算的上"全民Python"了.众所周知小学生的教材里面已经有Python ...

  2. JavaScript 新语法详解:Class 的私有属性与私有方法

    译者按: 为什么偏要用 # 符号? 原文:JavaScript's new #private class fields 译者:Fundebug 本文采用意译,版权归原作者所有 proposal-cla ...

  3. css控制元素高度自适应

    可以采用元素定位 + padding 的方式使特定元素高度自适应. css 样式: html,body{ height:100%; margin:; padding:; } .wrap { heigh ...

  4. MySQL中的自适应哈希索引

    众所周知,InnoDB使用的索引结构是B+树,但其实它还支持另一种索引:自适应哈希索引. 哈希表是数组+链表的形式.通过哈希函数计算每个节点数据中键所对应的哈希桶位置,如果出现哈希冲突,就使用拉链法来 ...

  5. 吴恩达机器学习笔记61-应用实例:图片文字识别(Application Example: Photo OCR)【完结】

    最后一章内容,主要是OCR的实例,很多都是和经验或者实际应用有关:看完了,总之,善始善终,继续加油!! 一.图像识别(店名识别)的步骤: 图像文字识别应用所作的事是,从一张给定的图片中识别文字.这比从 ...

  6. 在macos上基于python2.7安装PyQt5

    在python3上面安装PyQt5是十分简单的,可是,在python2.7上安装这个东西,着实让人折腾了一把.要总结一下,年纪大了,记性不好. 首先要安装最新版的Qt和python2,命令如下: br ...

  7. Redis in .NET Core 入门:(1) 安装和主要功能简介

    Redis(https://redis.io/), 是一个内存中的数据结构存储系统,它可以用作数据库.缓存和消息中间件. 安装Redis 我很少在开发机中直接装各种数据库,我一般使用Docker,针对 ...

  8. Docker 查看镜像信息

    欢迎关注个人微信公众号: 小哈学Java, 文末分享阿里 P8 资深架构师吐血总结的 <Java 核心知识整理&面试.pdf>资源链接!! 文章首发个人网站: https://ww ...

  9. IntentActionUtil【Intent的常见作用的工具类】

    版权声明:本文为HaiyuKing原创文章,转载请注明出处! 前言 主要用于通过Intent调用手机本地软件打开文件(doc.xsl.pdf.ppt.mp3.mp4等格式).安装apk.发送邮件.拨打 ...

  10. Nginx 简单记录

    Nginx("engine x")是一款是由俄罗斯的程序设计师Igor Sysoev所开发高性能的 Web和 反向代理 服务器,也是一个 IMAP/POP3/SMTP 代理服务器. ...