网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > 数据库 > SQL技巧
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
数据库:数据库教程,数据库技巧,Oracle教程,MySQL教程,Sybase教程,Access教程,DB2教程,数据库安全,数据库文摘
本月文章推荐
.存储过程使用技巧.
.关于对sql2000查询结果进行相关度.
.分页查询的一个帮助类.
.两台SQLServer数据同步解决方案(.
.SQL SERVER 2005 EXPRESS不能远程.
.带你轻松接触一个数据库的备份和.
.如何删除数据库中的冗余数据(翻.
.教你轻松恢复/修复SQL Server的M.
.SQL Server 2005密码安全追踪与存.
.角色赋给指定用户即可拥有备份数.
.SQL订阅状态的自动检查及自动启动.
.全面解析关系数据模型存在的不足.
.SQL Server 7.0 一般问题.
.SQL Server 索引结构及其使用(一.
.讲解SQL Server图像与大文本的输.
.关于SQL Server 2000对XML支持的.
.有助于数据创建安全环境的重要功.
.根据SQL存储过程名取得存储过程的.
.SQLSERVER扩展存储过程XP_CMDSHE.
.SQL Server的链接服务器技术小结.

如何正确的使用or展开来改写SQL查询

发表日期:2008-3-26


问题:

下面的这条sql应该怎么优化?

select * from sys_user 
where user_code = 'zhangyong' 
or user_code in 
(select grp_code 
from sys_grp 
where sys_grp.user_code = 'zhangyong')


Execution Plan
----------------------------------------------------------
0  SELECT STATEMENT Optimizer=RULE
1  0   FILTER
2  1     TABLE ACCESS (FULL) OF 'SYS_USER'
3  1     INDEX (UNIQUE SCAN) OF 'PK_SYS_GRP' (UNIQUE)


Statistics
----------------------------------------------------------
14  recursive calls
4  db block gets
30590 consistent gets
0  physical reads
0  redo size
1723  bytes sent via SQL*Net to client
425  bytes received via SQL*Net from client
2  SQL*Net roundtrips to/from client
0  sorts (memory)
0  sorts (disk)
3  rows processed

里面的查询返回的记录数一般只有一两条,但sys_user表的数据很多,怎么样才能让这条sql以sys_grp为驱动表?

表中记录情况如下:

SQL> select count(*) from sys_grp;
COUNT(*)----------25130
SQL> select count(*) from sys_user;
COUNT(*)
----------
15190

优化:

降低逻辑读是优化SQL的基本原则之一

我们尝试通过降低逻辑读来加快SQL的执行.

这里我们使用or展开来改写SQL查询:

select * from sys_user where user_code = 'zhangyong' 
union all
select * from sys_user where user_code <> 'zhangyong' 
and user_code in (select grp_code from sys_grp 
where sys_grp.user_code = 'zhangyong')

Statistics
----------------------------------------------------------
          0  recursive calls
          0  db block gets
         130 consistent gets
          0  physical reads
          0  redo size
       1723  bytes sent via SQL*Net to client
        425  bytes received via SQL*Net from client
          2  SQL*Net roundtrips to/from client
          1  sorts (memory)
          0  sorts (disk)
          3  rows processed

Execution Plan
----------------------------------------------------------
0      SELECT STATEMENT Optimizer=RULE
1    0   UNION-ALL
2    1     TABLE ACCESS (BY INDEX ROWID) OF 'SYS_USER'
3    2       INDEX (UNIQUE SCAN) OF 'PK_SYS_USER' (UNIQUE)
4    1     NESTED LOOPS
5    4       VIEW OF 'VW_NSO_1'
6    5         SORT (UNIQUE)
7    6           TABLE ACCESS (BY INDEX ROWID) OF 'SYS_GRP'
8    7             INDEX (RANGE SCAN) OF 'FK_SYS_USER_CODE' (NON-UNIQUE)
9    4       TABLE ACCESS (BY INDEX ROWID) OF 'SYS_USER'
10    9         INDEX (UNIQUE SCAN) OF 'PK_SYS_USER' (UNIQUE)

我们注意到,通过改写,逻辑读减少到130,从30590到130这是一个巨大的提高,减少逻辑读最终会减少资源消耗,提高SQL的执行效率.

这个改写把Filter改为了Nest LOOP,索引得以充分利用.从而大大提高了性能.

我们同时注意到,这里引入了一个排序

排序来自于这一步:

----------------------------------
6  5  SORT (UNIQUE)
7  6    TABLE ACCESS (BY INDEX ROWID) OF 'SYS_GRP'
8  7   INDEX (RANGE SCAN) OF 'FK_SYS_USER_CODE' (NON-UNIQUE)

----------------------------------

注释:在'SYS_GRP'表中,user_code 是非唯一键值,在in值判断里,要做sort unique排序,去除重复值,这里的union all是不需要排序的。

上一篇:如何用dbms_rowid获取rowid的详细信息 人气:1126
下一篇:轻松掌握数据库的具体分类及其相关概念 人气:1046
浏览全部SQL查询的内容 Dreamweaver插件下载 网页广告代码 祝你圣诞节快乐 2009年新年快乐