TempDB--临时表的缓存

9/1/2015来源:SQL技巧人气:2515

TempDB--临时表的缓存

--==========================================================================

在博客园看到一篇文章《SQLServer Temp tables 数据疑问》,文章中问道在没有任何负载情况下,还有大量的临时表,这是为什么?

--==========================================================================

让我们来实验探索下

首先选择任何一个用户数据库,执行以下脚本:

CREATE PROCEDURE USP_TempTableTestASBEGIN    SET NOCOUNT ON;    CREATE TABLE #TB1(C11 INT,C22 INT)    INSERT INTO #TB1    SELECT 1,1    SELECT * FROM #TB1ENDGOEXEC USP_TempTableTest

按照通用的理解,存储过程中的临时表会在调用中创建,在存储过程调用结束后释放,存储过程执行结束后,我们不应该在tempdb中找到#TB1开头的临时表。

让我们来检查下

use tempdbgo--======================================--查看临时表的列SELECT OBJECT_NAME(OBJECT_ID) AS ObjName,* FROM SYS.all_columnsWHERE OBJECT_NAME(OBJECT_ID) LIKE '%#%'

运行以上代码,可以很容易找到:

列名C11和C22和我们存储过程中定于的临时表列名一样,只是换了个马甲而已,别告诉我换了马甲你们就不认识”它“咯

--================================================================

解释:

上面看到的#A2206DOC这个临时表并不是我们存储过程中使用到的#TB1,但是但是两者存在一定关联。想象一下,每次存储过程执行,都需要创建和释放临时表,而创建和释放临时表又涉及到修改很多系统表的数据,而如果缓存这个临时表,那么下一次调用存储过程时,就避免创建和释放临时表所照成的资源消耗,从而达到一定的性能提升。

专业解释:

SQL Server删除一个临时对象时,不移除该对象的条目,当再次使用时,就无须重新创建临时对象,SQL Server为临时对象缓存一个数据页和一个IAM页,并回收剩余页,如果临时表的大小超过8MB,回收会异步进行。

--==================================================================

问题1: 多个存储过程并发调用的时候怎么办?

当并发调用时,会生成多个类似的临时表”缓存“以供调用,并保证一个临时表”缓存“在一个时间点内只能被一个回话执行的存储过程访问到,在访问结束后,会清理该临时表”缓存“的数据,然后供下一个回话使用。

问题2: 什么样的临时表会被”缓存“

当然不是所有临时表都会被缓存,需要满足一定的条件:

1. 没有创建命名约束

2. 临时对象在创建后没有执行影响临时表的数据定义语言DDL操作(如创建索引和创建统计)

3. 没有使用动态SQL创建临时对象

4. 临时对象被创建在另一个对象的内部,如存储过程、触发器、用户自定义函数或临时对象是一个用户自定义表值函数的分会表

--=====================================================================

依旧妹子引狼(最近小忙,不能保证妹子质量,勿怪)