1.在生产环境中不要出现Select *

这一点我想大家已经是比较熟知了,这样的错误相信会犯的人不会太多。但我这里还是要说一下。

不使用Select *的原因主要不是坊间所流传的将*解析成具体的列需要产生消耗,这点消耗在我看来完全可以忽略不计。更主要的原因来自以下两点:

  • 扩展方面的问题
  • 造成额外的书签查找或是由查找变为扫描

扩展方面的问题是当表中添加一个列时,Select *会把这一列也囊括进去,从而造成上面的第二种问题。

而额外的IO这点显而易见,当查找不需要的列时自然会产生不必要的IO,下面我们通过一个非常简单的例子来比较这两种差别,如图1所示。

图1.*带来的不必要的IO

2.声明变量时指定长度

这一点有时候会被人疏忽,因为对于T-SQL来说,如果对于变量不指定长度,则默认的长度会是1.考虑下面这个例子,如图2所示。

图2.不指定变量长度有可能导致丢失数据

3.使用合适的数据类型

合适的数据类型首先是从性能角度考虑,关于这一点,我写过一篇文章详细的介绍过,有兴趣可以阅读:对于表列数据类型选择的一点思考,这里我就不再细说了

不要使用字符串类型存储日期数据,这一点也需要强调一些,有时候你可能需要定义自己的日期格式,但这样做非常不好,不仅是性能上不好,并且内置的日期时间函数也不能用了。

4.使用Schema前缀来选择表

解析对象的时候需要更多的步骤,而指定Schema.Table这种方式就避免了这种无谓的解析。

不仅如此,如果不指定Schema容易造成混淆,有时会报错。

还有一点是,Schema使用的混乱有可能导致更多的执行计划缓存,换句话说,就是同样一份执行计划被多次缓存,让我们来看图3的例子。

图3.不同的schema选择不同导致同样的查询被多次缓存

5.命名规范很重要

推荐使用实体对象+操作这种方式,比如Customer_Update这种方式。在一个大型一点的数据库会存在很多存储过程,不同的命名方式使得找到需要的存储过程变得很不方便。因此有可能造成另一种问题,就是重复创建存储过程,比如上面这个例子,有可能命名规范不统一的情况下又创建了一个叫UpdateCustomer的存储过程。

6.插入大量数据时,尽量不要使用循环,可以使用CTE,如果要使用循环,也放到一个事务中

这点其实显而易见。SQL Server是隐式事务提交的,所以对于每一个循环中的INSERT,都会作为一个事务提交。这种效率可想而知,但如果将1000条语句放到一个事务中提交,效率无疑会提升不少。

打个比方,去银行存款,是一次存1000效率高,还是存10次100?下面,根据吉日的要求,补个例子,见代码1.

CREATE TABLE dbo.TestInsert
(Number INT PRIMARY KEY
);
--循环插入,不给力,我的笔记本45秒
DECLARE @index INT;
SET @index = 1;WHILE @index <= 100000
BEGININSERT dbo.TestInsert(Number) VALUES( @index);SET @index = @index + 1;
END--放到一个事务中循环,略好,但也不是最好,我的笔记本1秒
BEGIN TRAN
DECLARE @index INT;
SET @index = 1;WHILE @index <= 100000
BEGININSERT dbo.TestInsert(Number) VALUES( @index);SET @index = @index + 1;
ENDCOMMIT--批量插入,10W行,显示0秒,有兴趣的同学改成100W行进行测试
INSERT dbo.TestInsert(Number)SELECT TOP (100000) rn = ROW_NUMBER() OVER(ORDER BY c1.[object_id])FROM sys.columns AS c1CROSS JOIN sys.columns AS c2CROSS JOIN sys.columns AS c3ORDER BY c1.[object_id];--CTE方式,和上面那种方式大同小异,也是批量插入,比如:
WITH cte AS(SELECT TOP (100000) rn = ROW_NUMBER() OVER(ORDER BY c1.[object_id])FROM sys.columns AS c1CROSS JOIN sys.columns AS c2CROSS JOIN sys.columns AS c3ORDER BY c1.[object_id]
)
INSERT dbo.TestInsert(Number) SELECT rn FROM cte

代码1.几种插入方式的比较

7.where条件之后尽量减少使用函数或数据类型转换

换句话说,WHERE条件之后尽量可以使用可以嗅探参数的方式,比如说尽量少用变量,尽量少用函数,下面我们通过一个简单的例子来看这之间的差别。如图4所示。

图4.在Where中使用不可嗅探的参数导致的索引查找

对于另外一些情况来说,尽量不要让参数进行类型转换,再看一个简单的例子,我们可以看出在Where中使用隐式转换代价巨大。如图5所示。

图5.隐式转换带来的性能问题

8.不要使用旧的连接方式,比如(from x,y,z)

可能导致效率低下的笛卡尔积,当你看到下面这个图标时,说明查询分析器无法根据统计信息估计表中的数据结构,所以无法使用Loop join,merge Join和Hash Join中的一种,而是使用效率地下的笛卡尔积。

>   这里我再补充一点,我说得是“可能”导致,因为上面这个查询可能作为中间结果或是子查询,当你忘写了where条件时,会是笛卡尔积。你在最终结果中再用where过滤,可能得到的结果一模一样,但是中间的过程却大不相同

所以,尽量使用Inner join的方式替代from x,y,z这种方式。

9.使用游标时,加上只读只进选项

首先,我的观点是:游标是邪恶的,尽量少用。但是如果一定要用的话,请记住,默认设置游标是可进可退的,如果你仅仅设置了

declare c cursorfor

这样的形式,那么这种游标要慢于下面这种方式。

 declare c cursorlocal static read_only forward_onlyfor…

所以,在游标只读只进的情况下,加上上面代码所示的选项。

10.有关Order一些要注意的事情

首先,要注意,不要使用Order by+数字的形式,比如图6这种。

图6.Order By序号

当表结构或者Select之后的列变化时,这种方式会引起麻烦,所以老老实实写上列名。

还有一种情况是,对于带有子查询和CTE的查询,子查询有序并不代表整个查询有序,除非显式指定了Order By,让我们来看图7。

图7.虽然在CTE中中有序,但显式指定Order By,则不能保证结果的顺序

转载于:https://www.cnblogs.com/toddzhang/p/3338149.html

有关T-SQL的10个好习惯相关推荐

  1. 书写SQL必养成的好习惯

    前言 每一个好习惯都是一笔财富,本文分SQL后悔药, SQL性能优化,SQL规范优雅三个方向,分享写SQL的21个好习惯. 1. 写完SQL先explain查看执行计划(SQL性能优化) 日常开发写S ...

  2. 妨碍你成为CCIE的10个不良习惯

    CCIE #15626列出了妨碍你成为CCIE的10个不良习惯. 10. Not creating and sticking to a study plan. 9. Inputting command ...

  3. 女人必知:10个好习惯 让老公不想出轨

    阅读提示:要知道,妻子这10个动作是征求了数百名老公的意见之后进行总结得出的,不仅效果显著,杀伤力强,最关键的是简单易行. 女人必知:10个好习惯 让老公不想出轨 1.老公累了,靠在沙发上睡了,妻子要 ...

  4. Red Gate系列之一 SQL Compare 10.4.8.87 Edition 数据库比较工具 完全破解+使用教程

    原文:Red Gate系列之一 SQL Compare 10.4.8.87 Edition 数据库比较工具 完全破解+使用教程 Red Gate系列之一 SQL Compare 10.4.8.87 E ...

  5. Sams Teach Yourself SQL in 10 Minutes, Third Edition

    版权声明:原创作品,允许转载,转载时请务必以超链接形式标明文章原始出版.作者信息和本声明.否则将追究法律责任. http://blog.csdn.net/topmvp - topmvp Sams Te ...

  6. 深圳联嵌科技推荐:10个小习惯严重影响健康

    深圳联嵌科技推荐:10个小习惯严重影响健康 深圳市联嵌科技有限公司成立于2013年,前身为深圳市三叶草电子科技有限公司.主营业务为基于ARM平台的linux.Android设备的技术开发.企业培训等, ...

  7. 网络管理员不惜代价都要避免的10个坏习惯

    每个企业网络都由传输和存储信息的设备组成,面临着日益增长的复杂网络威胁. 网络管理员通过实施端到端控制,在避免未经授权的访问.数据丢失.恶意软件和安全漏洞方面发挥关键作用. 但是,在这一过程中会有几个 ...

  8. 让我郁闷的不行的SQL Anywhere 10

    今天, 一个非常郁闷的一天. 因为遇到了让我郁闷的SQL Anywhere 10. 因为我一不小心安装了一下这个软件, 同时也一不小心卸载了这个软件, 也正是同时, 它也一不小心的导致我的VS 200 ...

  9. 极客新闻——11、程序员需要避免的10个坏习惯

    本文笔记全部来自<极客新闻>--新鲜的技术资讯.权威的趋势剖析.别样的技术洞察 每位开发人员在自己的职业生涯.学习经历中,都会"开发"出一些坏习惯. Traversy ...

最新文章

  1. 组复制官方翻译五、Group Replication Security
  2. System.Transactions事务超时设置
  3. 【bzoj3309】DZY Loves Math 莫比乌斯反演+线性筛
  4. fillstyle属性_html设置或返回用于填充绘画的颜色渐变或模式的属性fillStyle
  5. 国内Python最有钱途的方向,开发第二,它排第一!
  6. 深入浅出Node.js (2) - 模块机制
  7. 微信小程序 开发过程中遇到的坑(一)
  8. python简单选择排序_python 选择排序
  9. Android studio的监听器初学者要懂
  10. 如何将word中的对象怎么显示到工具栏_职场必备!Word实用技巧最全总结(五)...
  11. localhost 拒绝了我们的连接请求。_Zipkin请求链路日志聚合
  12. 数据预处理—5.box-cox变换及python实现
  13. 【项目知识点】Vue中实现扫描二维码获取信息
  14. pmp 第六版 模拟卷3疑难问题
  15. 河北省对口计算机打字试题,河北省对口升学计算机基础.ppt
  16. java 返回进度条_Java中的命令行进度条
  17. 计算机毕业设计jsp酒店管理系统
  18. 软件测试动态分析,静态分析工具和动态测试工具
  19. 服务器端查看图片库 eog display Xforwarding
  20. Vue3路由,VueX3,Vue3生命周期函数

热门文章

  1. Linux系统中如何关闭触摸鼠标
  2. Java文件非法字符
  3. C# WinForm只允许运行一个窗体实例
  4. JavaScript prototype整理(网上的三种理解)
  5. 富数据控件 GridView(定义列、格式化、样式)
  6. 后台开发经典书籍--大话设计模式
  7. golang中的redigo
  8. Nginx-01:内容目录
  9. C语言数组的一些运算*a,a+1,a+1,a+0
  10. 目标代码文件、可执行文件和库