关注我们获得更多内容

作者 | 邹建,资深数据库专家,精通各项 SQL Server 技术,具有丰富的管理、维护、优化能力以及业务应用经验。他一直热心于技术知识的分享、传播,持续活跃在 CSDN 和 MSDN 社区,曾多年蝉联 CSDN 论坛积分榜首。

此外,邹建还是 2004~2010、2013年度 MVP(微软最有价值专家) 获得者。著有《深入浅出 SQL Server 2005开发、管理与应用实例》《SQL Server 2000开发与管理应用实例》等畅销书。

在 SQL Server 查询中,不经意思的隐匿数据类型转换可能导致极大的查询性能问题,比如一个看起来没有任何问题简单的条件:WHERE c = N’x’ ,如果 c 的数据类型是 varchar,并且表中包含大量的数据,这个查询可能导致极大的性能开销,因为这个操作会导致列 c 的数据类型转换为 nvarchar与常量值匹配,在 SQL Server 2008 及之后的版本中,这种操作做了增强,一定程度上降低了性能开销,参考SQL Server 2008 处理隐式数据类型转换在执行计划中的增强 。

不过在实际应用中发现,这种增强有时候似乎没有起到作用,还是会存在很大的性能问题。

最近找时间做了一个测试,找出了一种可能的问题。


1. 创建一个测试表

USE tempdb GOCREATE TABLE _t(    c varchar(50) );CREATE INDEX IX_c ON _t( c );GO-- 加入 10000 条数据INSERT _tSELECT (9999 + id) FROM(    SELECT TOP 10000 id = ROW_NUMBER() OVER( ORDER BY GETDATE() )    FROM sys.all_columns a, sys.all_columns )ID

2. 通过执行计划看下查询计划

-- Rebuild索引,确保无索引碎片和统计信息准确

ALTER INDEX IX_c ON _t REBUILD;GO

SET SHOWPLAN_ALL ON

GO

SELECT * FROM _t WHERE c = N'10005b';

GO

SET SHOWPLAN_ALL OFF;

注意EstimateRows列,该列值为1,表示评估的满足条件的数据是1条,现在看起来一切正常 。

3.把数据变一下,将大量数据变成相同值

-- 将 5000 条数据值变成一样,重建索引之后重新测试

UPDATE _t SET c = '15000' WHERE c >= '15000'

ALTER INDEX IX_c ON _t REBUILD;

GO

SET SHOWPLAN_ALL ON

GO

SELECT * FROM _t WHERE c = N'10005';

GO

SET SHOWPLAN_ALL OFF;

然后我们发现评估的记录数变大了

4. 继续加大相同值的比例

-- 继续加大相同值的比例,重建索引之后重新测试

UPDATE _t SET c = '11000' WHERE c >= '11000' AND c < '15000'

ALTER INDEX IX_c ON _t REBUILD;

GO

SET SHOWPLAN_ALL ON

GO

SELECT * FROM _t WHERE c = N'10005';

GO

SET SHOWPLAN_ALL OFF;

GO

-- 继续加大相同值的比例,重建索引之后重新测试

UPDATE _t SET c = '10100' WHERE c >= '10100' AND c < '11000'

ALTER INDEX IX_c ON _t REBUILD;

GO

SET SHOWPLAN_ALL ON

GO

SELECT * FROM _t WHERE c = N'10005';

GO

SET SHOWPLAN_ALL OFF;

相应的,预估的行数也在增加

如果我们使用正确的数据类型,WHERE c = ‘10005’,则始终可以得到正确的预估行数。

我不确定 SQL Server是按照什么标准来预估这种情况下的记录数,从执行计划看,它将 nvarchar 值通过 GetRangeThroughConvert 评估出一个范围,实际执行的是一个范围 seek,在试验中,查询的值是一个常量,可以准确评估,难道这个转换之后,把常量当变量评估了,所以是一个泛泛的评估结果值。

这个问题看起来不大,但在实际应用中,如果表的数据量很大,并且不是平均分布的话,这种错误的预估结果带来的性能影响是很大的,比如明明满足条件的很少,可以 seek, 但评估的结果很大,执行计划变 Scan了,在复杂的执行计划中,这个带来的影响更大。

看起来,2008(包括R2)还没有那么省心,这种问题还得控制,特别是程序中,.Net过来的参数通常都是 nvarchar类型,这种导致性能问题的情况遇到N多了 。

最后啰嗦一下的是,在 SQL Server 2014中,没有再发现这个问题(不知道 2012中怎么样)


原创:邹建。

投稿:有投稿意向技术人请在公众号对话框留言。

转载:意向文章下方留言。

更多精彩请关注 “数据和云” 公众号 。

专业数据库服务:点击 “阅读原文” 。


招聘专栏

Oracle 售前工程师(广州、深圳、上海、武汉、北京、石家庄)

Oracle 高级工程师(上海、深圳、北京、成都、昆明、贵州、西宁)

MySQL 技术经理(上海、南京、成都)

MySQL 工程师(上海、杭州)

超高待遇:丰厚的年终奖,五险一金,高额学习基金,团建旅游,法定节假日,福利假期等。

推荐他人成功入职有好礼(iPhone X)相送 。

投递简历至邮箱:hr@enmotech.com

资源下载

关注公众号:数据和云(OraNews)回复关键字获取

2018DTCC , 数据库大会PPT

2017DTC,2017 DTC 大会 PPT

DBALIFE ,“DBA 的一天”海报

DBA04 ,DBA 手记4 电子书

122ARCH ,Oracle 12.2体系结构图

2017OOW ,Oracle OpenWorld 资料

PRELECTION ,大讲堂讲师课程资料

近期文章

仅仅使用AWR做报告? 性能优化还未入门

实战课堂:一则CPU 100%的故障分析

杨廷琨:如何编写高效SQL(含PPT)

一份高达555页的技术PPT会是什么样子?

大象起舞:用PostgreSQL解海盗分金问题

ProxySQL!像C罗一样的强大

高手过招:用SQL解决环环相扣刑侦推理问题

SQL Server 2008处理隐式数据类型转换在执行计划中的增强相关推荐

  1. MySQL 避坑指南之隐式数据类型转换

    作者 | 不剪发的Tony老师 责编 | 欧阳姝黎 出品 | CSDN博客 ????知之为知之,不知为不知,是知也.--<论语> 今天我们来聊聊 MySQL 中存在的隐式数据类型转换以及可 ...

  2. Sql Server函数全解三数据类型转换函数和文本图像函数

    原文:Sql Server函数全解<三>数据类型转换函数和文本图像函数 一:数据类型转换函数 在同时处理不同数据类型的值时,SQL Server一般会自动进行隐士类型转换.对于数据类型相近 ...

  3. SQL Server 2008使用LINQ进行数据访问(转载自IT168 [ http://www.it168.com/ ])

    SQL Server 2008使用LINQ进行数据访问 2008年09月23日 IT168网站原创 作者:微软中国 编辑:胡铭娅 评论:0条 本文Tag: LINQ SQL Server 2008 微 ...

  4. java取geosever数据_Geoserver - 连接到SQL Server 2008 Express并获取数据

    Bref history: - 我正在使用MapServer,一切都运行正常,除了当我使用PixMap直接在图层上绘制标记时,它在tile之间切割 . 因此,我决定更改为GeoServer,它看起来能 ...

  5. JS基础语法(05)-隐式数据类型转换

    1.为什么要有隐式类型转换? 当运算符在运算时,如果两边数据不统一,CPU就无法计算,这时我们编译器会自动将运算符两边的数据做一个数据类型转换,转成一样的数据类型再计算 2.什么是隐式类型转换:编译器 ...

  6. oracle隐式转换函数,隐式数据类型转换案例一则

    在OLTP系统中,要求对频繁执行的SQL使用绑定变量(唯一值少的字段,如type.status等,数据分布不均是常见情况,这种字段不建议使用绑定变量). 在使用绑定变量时,有时开发人员对使用的字段类型 ...

  7. 【JavaScript】-- 隐式数据类型转换

    数据类型的隐式转换 一.基本数据类型的转换 1.加减乘除 加法:如果字符串加数字,数字就会转成字符串.而 数字加数字 或 字符串加字符串 则不需要进行转换直接进行运算或字符串拼接. 减法: 数字减字符 ...

  8. Javascript隐式数据类型转换(==、+、Object)

    <!DOCTYPE html> <html><head><meta charset="utf-8"><title>< ...

  9. SQL Server 2008如何导出带数据的脚本文件

    SQL之前的版本导出脚本的时候只能把数据表的结构导出,却不能把其中的数据一齐导出来,在平时的应用中难免有时希望获取数据导出的脚本,比如有时SQL版本不一样的时候,无法把高版本的数据库备份直接还原到低版 ...

最新文章

  1. 弹出页(指定高度,自由拖动,点击空白包括状态栏触发)
  2. Linux学习之命令【1】
  3. 雷达波形样式_案例分析 | 基于XSIM的雷达电子对抗仿真
  4. P1060 开心的金明(01背包)
  5. VMware文件共享
  6. Java面向对象之抽象方法抽象类、接口的使用
  7. Linux 文件系统IO性能优化
  8. 系统自动化制作工资条,很简单,还能发送短信息
  9. php编写出一个时钟,用HTML5实现一个时钟
  10. zoj 1005 Jugs BFS
  11. 数据标注软件 【labelimg】
  12. Oracle(11g)数据库安装详细图解教程
  13. LPC1768 双通道ADC采集数据并发送
  14. 线稿图视频制作补充(调整视频亮度、对比度;删除无用文件)
  15. 一个简单的学籍信息管理系统,基于PHP和Bootstrap的实现
  16. python计算模型psi_模型稳定度指标PSI与IV
  17. 125家单位联合完成微生物组实验手册(Microbiome Protocol eBook)第1版
  18. MyEclipse2017破解时 ACTIVATION_KEY为null
  19. 优思学院|利用Excel进行双样本t检验
  20. App 自动化解决方案 [开源项目] 基于 Appium 的 UI 自动化测试框架完美版

热门文章

  1. vm和openstack_最佳新OpenStack技巧和窍门
  2. 2014年图灵奖_2014年人民选择奖:投下您的一票
  3. (4)Node.js 安装
  4. jQuery 版本viewer.js插件的结构分析与学习
  5. ssm 转发请求_千呼万唤!阿里内部终于把这份SSM框架技术学习笔记分享出来了...
  6. 视觉SLAM笔记(61) 单目稠密建图
  7. 视觉SLAM笔记(26) 状态估计问题
  8. 职称计算机承诺书,职称申报承诺书范本
  9. 打印异常堆栈_关于日志打印行号的性能案例
  10. file上传代码 ios_文件上传_iOS上传SDK_客户端上传_上传SDK_视频点播 - 阿里云