原文:SQL Server中语句的自动参数化



  1. use master
  2. go
  3. if exists(select * from sys.databases where name = 'test')
  4. drop database test
  5. go
  6. --创建数据库
  7. create database test
  8. if exists(select * from sys.tables where name = 't')
  9. drop table t
  10. go
  11. --1.创建表t
  12. create table t(i int);
  13. --2.添加100000条记录,消耗56秒
  14. declare @i int;
  15. declare @sql varchar(1000);
  16. set @i = 1
  17. set @sql = '';
  18. while @i <= 100000
  19. begin
  20. set @sql = 'insert into t values(' + cast(@i as varchar) +')'
  21. begin tran
  22. exec(@sql)
  23. commit tran
  24. set @i = @i + 1
  25. end
  26. --3.1查询的统计信息
  27. --但发现没有返回记录
  28. select st.text,
  29. SUBSTRING(st.text, (qs.statement_start_offset/2)+1,
  30. ((CASE qs.statement_end_offset
  31. WHEN -1 THEN DATALENGTH(st.text)
  32. ELSE qs.statement_end_offset
  33. END - qs.statement_start_offset)/2) + 1) AS statement_text,
  34. qs.*
  35. from sys.dm_exec_query_stats qs
  36. cross apply sys.dm_exec_sql_text(qs.sql_handle) st
  37. where cast(st.text as varchar) like '%insert into t values%'
  38. --3.2查询的统计信息
  39. --有返回,说明SQL Server在执行动态生成的语句时,已经参数化了
  40. --execution_count字段的值为100000
  41. select st.text,
  42. SUBSTRING(st.text, (qs.statement_start_offset/2)+1,
  43. ((CASE qs.statement_end_offset
  44. WHEN -1 THEN DATALENGTH(st.text)
  45. ELSE qs.statement_end_offset
  46. END - qs.statement_start_offset)/2) + 1) AS statement_text,
  47. qs.*
  48. from sys.dm_exec_query_stats qs
  49. cross apply sys.dm_exec_sql_text(qs.sql_handle) st
  50. where cast(st.text as varchar) like '%insert into%'
  51. --3.3 查看数据库是否强制参数化
  52. --并没有强制参数化,说明上面是系统自动对语句进行参数化了
  53. select name,
  54. is_parameterization_forced --返回0,说明并没有强制参数
  55. from sys.databases
  56. where name = 'test'
  57. --3.4 再次执行后,再次执行3.2后发现,execution_count为100001
  58. insert into t values(100001)
  59. --3.5 这次在语句中加了空格,execution_count为100002
  60. insert into t values( 100002 )
  61. --3.5 这次在语句中加了架构
  62. --返回2条记录,一条为100002,另一个为1
  63. --虽然2条数据的sql_handle、plan_handle都不相同,但是query_hash、query_plan_hash相同
  64. insert into dbo.t values( 100003 )
  65. --3.6 显示缓存的执行计划
  66. --返回2条数据,一条为100002,一个为1
  67. --缓存对象类型为编译计划,对象类型为Prepared
  68. select *
  69. from sys.dm_exec_cached_plans cp
  70. cross apply sys.dm_exec_sql_text(cp.plan_handle) st
  71. where st.text like '%insert into%'

最后总结一下:

1.对于一些比较简单的语句,SQL Server能自动进行参数化,会自动忽略空格。

2.但当语句有不同时,比如加了对象所属的架构后,系统就没办法进行参数化了,但很有意思的是query_hash和query_plan_hash都相同,也就是说明本质上这些查询都是一样的。

所以可以通过group by这些hash值,计算编译了多少次。

发布了416 篇原创文章 · 获赞 135 · 访问量 94万+

SQL Server中语句的自动参数化的更多相关文章

  1. 使用sql语句创建修改SQL Server标识列(即自动增长列)

    一.标识列的定义以及特点SQL Server中的标识列又称标识符列,习惯上又叫自增列.该种列具有以下三种特点:1.列的数据类型为不带小数的数值类型2.在进行插入(Insert)操作时,该列的值是由系统 ...

  2. SQL Server中常用的SQL语句(转):

    SQL Server中常用的SQL语句 转自:http://www.cnblogs.com/rainman/archive/2013/05/04/3060428.html 1.概述 名词 笛卡尔积.主 ...

  3. SQL Server中的SQL语句优化与效率问题

    很多人不知道SQL语句在SQL SERVER中是如何执行的,他们担心自己所写的SQL语句会被SQL SERVER误解.比如: select * from table1 where name='zhan ...

  4. SQL Server中常用的SQL语句

    1.概述 名词 笛卡尔积.主键.外键 数据完整性 实体完整性:主属性不能为空值,例如选课表中学号和课程号不能为空 参照完整性:表中的外键取值为空或参照表中的主键 用户定义完整性:取值范围或非空限制,例 ...

  5. SQL Server中CURD语句的锁流程分析

    我只在数据库选项已开启“行版本控制的已提交读”(READ_COMMITTED_SNAPSHOT为ON)中进行了观察. 因此只适用于这种环境的数据库. 该类数据库支持四种不同事务隔离级别,下面分别观察数 ...

  6. SQL Server中怎么查看每个数据库的日志大小,以及怎么确定数据库的日志文件,怎么用语句收缩日志文件

    一,找到每个数据库的日志文件大小 SQL Server:查看SQL日志文件大小命令:dbcc sqlperf(logspace) DBA 日常管理工作中,很重要一项工作就是监视数据库文件大小,及日志文 ...

  7. (4.19)sql server中的事务模式(隐式事务,显式事务,自动提交事务)

    (4.19)sql server中的事务模式(隐式事务,显式事务,自动提交事务) 1.概念:隐式事务,显式事务,自动提交事务 2.操作:如何设置事务模式 3.存储过程中的事务 XACT_ABORT 1 ...

  8. SQL Server中的SQL语句优化与效率

    很多人不知道SQL语句在SQL SERVER中是如何执行的,他们担心自己所写的SQL语句会被SQL SERVER误解.比如: select * from table1 where name='zhan ...

  9. SQL Server中一些有用的日期sql语句

    SQL Server中一些有用的日期sql语句 1.一个月第一天的 SELECT DATEADD(mm, DATEDIFF(mm,0,getdate()), 0) 2.本周的星期一 SELECT DA ...

随机推荐

  1. These dependencies were not found: *!!vue-style-loader!css-loader?

    在vue中使用less首先要下载依赖:npm install less less-loader --save-dev 下载好之后就可以.vue文件中使用lang="less"和@i ...

  2. Cross-Site Request Forgery (CSRF)

    https://www.owasp.org/index.php/Cross-Site_Request_Forgery_(CSRF) Overview Cross-Site Request Forger ...

  3. unittest 的用法

    一.discover方法 discover方法可以根据标准加载用例,并将结果返回给测试套件(suite),start_dir:待测试的目录,pattern:测试用例文件名的匹配规. 如: start_ ...

  4. 几种主流浏览器内置http抓包工具软件使用方

    对于学习网站的人或者相关编程人员,经常需要用到http抓包工具来跟踪网页,但主流抓包软件如httpwatch.httpanalyzerstdv都是收费的,破解版往往也不稳定.实际上现在很多浏览器都内置 ...

  5. Facebook libra开发者文档- 3 -Life of a Transaction交易生命周期

    Life of a Transaction交易的生命周期 https://developers.libra.org/docs/life-of-a-transaction 为了更深入地了解Libra交易 ...

  6. 自定义基于IFC数据的施工进度数据结构

    <DataSource>D:/qlbz20190802.ifc</DataSource> <Datas> <Data> <ID></I ...

  7. docker之redis使用

    #拉取redis > docker pull redis:latest latest: Pulling from library/redis 8d691f585fa8: Pull complet ...

  8. PAT 甲级 1054 The Dominant Color (20 分)(简单题)

    1054 The Dominant Color (20 分)   Behind the scenes in the computer's memory, color is always talked ...

  9. LeetCode_66. Plus One

    66. Plus One Easy Given a non-empty array of digits representing a non-negative integer, plus one to ...

  10. mysql创建数据库、表、写数据

    mysql> create database ceshi; Query OK, 1 row affected (0.01 sec) 给数据库授权,否则程序时无法连接ceshi数据库的,每次创建一 ...