"5第六掌去掉Where子句中的ISNULL"> "5第六掌去掉Where子句中的ISNULL" />
欢迎来到天天文库
浏览记录
ID:8980667
大小:75.00 KB
页数:10页
时间:2018-04-13
《sql语句优化之降龙十八掌》由会员上传分享,免费在线阅读,更多相关内容在应用文档-天天文库。
1、基于索引的SQL语句优化之降龙十八掌1前言22总纲23降龙十八掌3第一掌避免对列的操作3第二掌避免不必要的类型转换4第三掌增加查询的范围限制4第四掌尽量去掉"IN"、"OR"4第五掌尽量去掉"<>"5第六掌去掉Where子句中的ISNULL和ISNOTNULL5第七掌索引提高数据分布不均匀时查询效率5第八掌利用HINT强制指定索引6第九掌屏蔽无用索引6第十掌分解复杂查询,用常量代替变量7第十一掌like子句尽量前端匹配7第十二掌用Case语句合并多重扫描7第十三掌使用nls_date_format8第十四掌使用基于函数的索引8第十五掌基于函数的索引要求等
2、式匹配9第十六掌使用分区索引9第十七掌使用位图索引9第十八掌决定使用全表扫描还是使用索引94总结101前言客服业务受到SQL语句的影响非常大,在规模比较大的局点,往往因为一个小的SQL语句不够优化,导致数据库性能急剧下降,小型机idle所剩无几,应用服务器断连、超时,严重影响业务的正常运行。因此,称低效的SQL语句为客服业务的‘恶龙’并不过分。数据库的优化方法有很多种,在应用层来说,主要是基于索引的优化。本次秘笈根据实际的工作经验,在研发原来已有的方法的基础上,进行了一些扩充,总结了基于索引的SQL语句优化的降龙十八掌,希望有一天你能用其中一掌来驯服客服
3、业务中横行的‘恶龙’。2总纲l建立必要的索引这次传授的降龙十八掌,总纲只有一句话:建立必要的索引,这就是后面降龙十八掌的内功基础。这一点看似容易实际却很难。难就难在如何判断哪些索引是必要的,哪些又是不必要的。判断的最终标准是看这些索引是否对我们的数据库性能有所帮助。具体到方法上,就必须熟悉数据库应用程序中的所有SQL语句,从中统计出常用的可能对性能有影响的部分SQL,分析、归纳出作为Where条件子句的字段及其组合方式;在这一基础上可以初步判断出哪些表的哪些字段应该建立索引。其次,必须熟悉应用程序。必须了解哪些表是数据操作频繁的表;哪些表经常与其他表进行
4、连接;哪些表中的数据量可能很大;对于数据量大的表,其中各个字段的数据分布情况如何;等等。对于满足以上条件的这些表,必须重点关注,因为在这些表上的索引,将对SQL语句的性能产生举足轻重的影响。不过下面还是总结了一下降龙十八掌内功的入门基础,建立索引常用的规则如下:1、表的主键、外键必须有索引;2、数据量超过300的表应该有索引;3、经常与其他表进行连接的表,在连接字段上应该建立索引;4、经常出现在Where子句中的字段,特别是大表的字段,应该建立索引;5、索引应该建在选择性高的字段上;(字段重复度低?)6、索引应该建在小字段上,对于大的文本字段甚至超长字段
5、,不要建索引;7、复合索引的建立需要进行仔细分析;尽量考虑用单字段索引代替:A、正确选择复合索引中的主列字段,一般是选择性较好的字段;B、复合索引的几个字段是否经常同时以AND方式出现在Where子句中?单字段查询是否极少甚至没有?如果是,则可以建立复合索引;否则考虑单字段索引;C、如果复合索引中包含的字段经常单独出现在Where子句中,则分解为多个单字段索引;D、如果复合索引所包含的字段超过3个,那么仔细考虑其必要性,考虑减少复合的字段;E、如果既有单字段索引,又有这几个字段上的复合索引,一般可以删除复合索引;8、频繁进行数据操作的表,不要建立太多的索
6、引;9、删除无用的索引,避免对执行计划造成负面影响;以上是一些普遍的建立索引时的判断依据。一言以蔽之,索引的建立必须慎重,对每个索引的必要性都应该经过仔细分析,要有建立的依据。因为太多的索引与不充分、不正确的索引对性能都毫无益处:在表上建立的每个索引都会增加存储开销,索引对于插入、删除、更新操作也会增加处理上的开销。另外,过多的复合索引,在有单字段索引的情况下,一般都是没有存在价值的;相反,还会降低数据增加删除时的性能,特别是对频繁更新的表来说,负面影响更大。1降龙十八掌第一掌避免对列的操作任何对列的操作都可能导致全表扫描,这里所谓的操作包括数据库函数、
7、计算表达式等等,查询时要尽可能将操作移至等式的右边,甚至去掉函数。例1:下列SQL条件语句中的列都建有恰当的索引,但30万行数据情况下执行速度却非常慢:select*fromrecordwheresubstrb(CardNo,1,4)='5378'(13秒)select*fromrecordwhereamount/30<1000(11秒)select*fromrecordwhereto_char(ActionTime,'yyyymmdd')='19991201'(10秒)由于where子句中对列的任何操作结果都是在SQL运行时逐行计算得到的,因此它不得不
8、进行表扫描,而没有使用该列上面的索引;如果这些结果在查询编译时就能得到,那么就可
此文档下载收益归作者所有