首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 数据库 > oracle >

升星,散分,来者有分!写好SQL(1),该如何解决

2012-02-15 
升星,散分,来者有分!写好SQL(1)写好SQL一、基本SQL优化本节谈的主要是基于RBO(RULE BASED OPTIMIZER)的SQL

升星,散分,来者有分!写好SQL(1)
写好SQL
一、基本SQL优化
本节谈的主要是基于RBO(RULE BASED OPTIMIZER)的SQL 优化
Select * 和select 字段1,字段2。这两种写法,前者对写的人来说比较简单,后者则要多点点东西,而对于性能来,一般都是建议写具体字段的,ORACLE在解析的过程中, 会将‘*’ 依次转换成所有的列名, 这个工作是通过查询数据字典完成的, 这意味着将耗费更多的时间。
From 表1 别名,表2 别名。一般来说基于rbo的方式,ORACLE的解析器按照从右到左的顺序处理FROM子句中的表名,因此FROM子句中写在最后的表(基础表 driving table)将被最先处理。 在FROM子句中包含多个表的情况下,你必须选择记录条数最少的表作为基础表。当ORACLE处理多个表时, 会运用排序及合并的方式连接它们。首先,扫描第一个表(FROM子句中最后的那个表)并对记录进行派序,然后扫描第二个表(FROM子句中最后第二个表),最后将所有从第二个表中检索出的记录与第一个表中合适记录进行合并。另外给表名加上简洁的别名,也是不错的习惯,一是在多表关联时写起来方便,减少写错的机会,另一方面解析执行的时候,也相对快一点。
Where 条件1 and 条件2。ORACLE采用自下而上的顺序解析WHERE子句,根据这个原理,表之间的连接必须写在其他WHERE条件之前, 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾。
GROUP BY 提高GROUP BY 语句的效率, 可以通过将不需要的记录在GROUP BY 之前过滤掉。
  order by 字段名或者order 1(列),2(列)。Order by是比较费时间的操作,所以尽可能在语句减少各子句内的使用,也尽可能在索引列上进行排序。
在写完SQL语句,对于那些数据量比较大的语句,应该多用下执行计划来分析下语句,看看语句是否可以有优化的地方,一般来说,最常见的SQL语句慢,是因为进行了全表扫描,在进行WHERE过滤数据时,有索引列的就尽可能在索引列上过滤,这样会快很多,也尽可能避免写使索引列过滤失效的语句。在数据比较的语句中,如果类似于>10之类的,尽可能用>=11来代替,两者的区别在于,>10的操作,需要先找到=10的,然后再开始找>10的第一条纪录,而后者是直接跳到11这条纪录开始。

二、Rownum和row_number() over()的使用
ROWNUM是oracle从8开始提供的一个伪列,是把SQL出来的结果进行编号,始终从1开始,常见的用途就是用来分页输出.
比如 

SQL code
SELECT *          FROM torderdetail a     WHERE ROWNUM <= 10 

这条语句就是输出前10条纪录,在这里用途上类似于sql sever的top,不过rownum对于指定编号区间的输出应该说更强大
SQL code
SELECT *          FROM (SELECT a.*, ROWNUM rn              FROM torderdetail a)     WHERE rn >= 10 AND rn <= 20


这条语句即是输出第10到第20条纪录,这里之所以用rownum rn,是把rownum转成实例,因为rownum本身只能用<=的比较方式,只有转成实列,这样就可做 >=的比较了。
在实际用途中,常常会要求取最近的几条纪录,这就需要先对纪录进行排序后再取rownum<=
一般常见的
SQL code
SELECT *      FROM (SELECT   a.*            FROM torderdetail a        ORDER BY order_date DESC) WHERE ROWNUM <= 10 


而在CSDN曾经发生过讨论,关于取近的10条纪录,有人给出这样的语句
SQL code
SELECT   a.*            FROM torderdetail a           WHERE ROWNUM <= 10ORDER BY order_date DESC 


之所以会出现这样的语句,主要是从效率上的考虑,前面条语句,是要进行全表扫描后再排序,然后再取10条纪录,后一条语句则不会全表扫描,只会取出10条纪录,很明显后条语句的效率会高许多。
那为什么会有争议呢,那就在于在执行顺序上争议,是先执行排序取10条纪录,还是取10条纪录,再排序呢?两种顺序取出来的结果是截然相反的,先排序再取10条,就是取最近的10条,而先取10条,再排序,则取出的最早的10条纪录。对于此语句,普遍的认为执行顺序是先取10条纪录再排序的。所以此语句应该是错误。但实际上并非如此,此语句的执行顺序和order by的字段有关系,如果你order by 的字段是pk,则是先排序,再取10条(速度比第一种语句快),而排序字段不是PK 时,是先取10条再排序,此时结果就与要求不一样了,所以第二种写法一定要在排序字段是主键的情况下才能保证结果正确。
Row_number() over()这个分析函数是从9I开始提供的,一般的用途和rownum差不多。
一般写法row_number() over( order by order_date desc) 生成的顺序和rownum的语句一样,效率也一样(对于同样有order by 的rownum语句来说),所以在这种情况下两种用法是一样的。
而对于分组后取最近的10条纪录,则是rownum无法实现的,这时只有row_number可以实现,row_number() over(partition by 分组字段 order by 排序字段)就能实现分组后编号,比如说要取近一个月的每天最后10个订单纪录
SQL code
SELECT *          FROM (SELECT a.*,ROW_NUMBER () OVER (PARTITION BY TRUNC (order_date) ORDER BY order_date DESC)                                                                           rn          FROM torderdetail a) WHERE rn <= 10 


Rownum的另类用法,有时候我们会遇到这种需求,要求输出当月的所有天数,许多人会烦恼,数据库里又没有这样的表,怎么输出一个月的所有天数呢?用rownum就能解决:
SQL code
SELECT     TRUNC (SYSDATE, 'MM') + ROWNUM - 1      FROM DUALCONNECT BY ROWNUM <= TO_NUMBER (TO_CHAR (LAST_DAY (SYSDATE), 'dd')) 



------解决方案--------------------


接分来了
[解决办法]





==== 思想重于技巧 ====

[解决办法]
十分感谢
[解决办法]
谢谢LZ
[解决办法]
接分啊!
順便問一下,技術積分要多少分才可以升星啊?
[解决办法]
经典。
支持楼主。
[解决办法]
呵呵,来接分了
[解决办法]
lz真有意思,恭喜了!
同时也谢谢在这里发一些学习资料!共勉!
[解决办法]
mark
[解决办法]
恩,楼主狠强大

向楼主学习
[解决办法]
我已经准备好了,你准备好了吗?
[解决办法]

探讨
引用:
接分啊!
順便問一下,技術積分要多少分才可以升星啊?

5000分啊,呵呵,我用了8个半月混到5000分,不过最近的3000分比较快,只用了一个月,多谢大家棒场

[解决办法]
刚又看了遍,真的很好

‘而在CSDN曾经发生过讨论,关于取近的10条纪录’ 关于这个也没楼主想的那么全面

学习学习
[解决办法]
楼主是不是会法语?
“C'est la Vie”
某曾经还学过一点呢,呵呵!共勉!
[解决办法]
很好,很强大。
支持hebo,。
[解决办法]
即接分,又学习了,多谢楼主了!
[解决办法]
ding
[解决办法]
jf
[解决办法]
支持楼主!努力升星----
[解决办法]
厉害
[解决办法]
赫赫,恭喜楼主了!
[解决办法]
谢谢总结这么好!!
[解决办法]
gx,

[解决办法]
学习一下 :)
[解决办法]
接分买裤衩~
[解决办法]
来接分。。。
发现从2个角到3个角也好慢。。哎
[解决办法]
高手!
[解决办法]
Congratulations~
[解决办法]
谢谢了,接分来啦,楼住真有才
[解决办法]
GXGX~
[解决办法]
up
[解决办法]
恭喜
恭喜
------解决方案--------------------


学习一下
[解决办法]
jiefen
[解决办法]
ding
[解决办法]
jf
[解决办法]
我什么时候能升星呀
[解决办法]
up
[解决办法]
谢谢LZ~~
[解决办法]
接分
[解决办法]
楼主对sql的分析使我受益颇多!
[解决办法]
接分
[解决办法]
能者为师
[解决办法]

探讨
lz真有意思,恭喜了!
同时也谢谢在这里发一些学习资料!共勉!

[解决办法]
恭喜.
学习.
[解决办法]
探讨
恭喜.
学习.

[解决办法]
make
[解决办法]
恭喜,楼主好厉害啊!
学习
[解决办法]
拿分来~~~
[解决办法]
恭喜
MARK
[解决办法]
jie fen
[解决办法]
恭喜,jf
[解决办法]
已阅

偶是初学者,学习下

照顶顶......
[解决办法]
恭喜!~希望樓主能加到我的Oracle群里交流一下經驗,對大家都很有幫助!~ 

群號:62026713

再次恭喜!!
[解决办法]
Rownum的另类用法,有时候我们会遇到这种需求,要求输出当月的所有天数,许多人会烦恼,数据库里又没有这样的表,怎么输出一个月的所有天数呢?用rownum就能解决:

SQL code
SELECT TRUNC (SYSDATE, 'MM') + ROWNUM - 1
FROM DUAL
CONNECT BY ROWNUM <= TO_NUMBER (TO_CHAR (LAST_DAY (SYSDATE), 'dd'))


CONNECT BY ROWNUM <= TO_NUMBER (TO_CHAR (LAST_DAY (SYSDATE), 'dd')) 里面加了=将下个月的第一天都取进来了
应该是CONNECT BY ROWNUM < TO_NUMBER (TO_CHAR (LAST_DAY (SYSDATE), 'dd'))
[解决办法]
CONNECT BY ROWNUM <= TO_NUMBER (TO_CHAR (LAST_DAY (SYSDATE), 'dd')) 
好像在9i里用不了
[解决办法]
接分
[解决办法]
顶.
[解决办法]
探讨


[解决办法]
接分
[解决办法]
不够
------解决方案--------------------


最后再顶一下
[解决办法]
学习
[解决办法]
顶LZ ,攒点分
[解决办法]
呵呵,接分,顺便学些,正在学oracle!
[解决办法]
JING
[解决办法]
jf
[解决办法]
顶一个
[解决办法]
学习
[解决办法]
接分,我也想要颗星
[解决办法]
我接不住了 忽忽~~
[解决办法]
接分~~~~~~~~~~~~~
[解决办法]
好东西,收藏了!
[解决办法]
真的很好哦
[解决办法]
接分
[解决办法]
楼主是强人阿!
这里人气这么旺!
顶一下!
[解决办法]
恭喜楼主
接分,呵呵
[解决办法]
学习中……
[解决办法]
LZ就是厉害,我经常看到你回答问题,顶下。顺便接点分,你这么多给我点也没事哈。呵呵
[解决办法]
先顶再读!
[解决办法]
学习到了!
[解决办法]
支持hebo,支持散分,散分万岁,不散分可耻。。。

BTW:CSDN是不是不能发表情的哦,刚刚来到。。。
[解决办法]
顶一下 走你~
[解决办法]
什么都不说,只接分……
[解决办法]
up!
[解决办法]
JF
[解决办法]
学习中
[解决办法]
楼主好强大

热点排行