网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
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!
当前位置 > 网站建设学院 > 网络编程 > 数据库 > Oracle教程
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
数据库:数据库教程,数据库技巧,Oracle教程,MySQL教程,Sybase教程,Access教程,DB2教程,数据库安全,数据库文摘
本月文章推荐
.Oracle电子商务套件11i10首次亮相.
.在linux8.0下,oracle9i的配置说.
.Oracle基本数据类型存储格式浅析.
.SQL Server和Oracle数据锁定比较.
.解决JOB的Interval输入参数过长的.
.关于block中行数据的存储与空间重.
.集合和成员函数.
.TCP/IP远程访问操作.
.Oracle数据字典与动态性能表的相.
.Oracle数据库SQLPLUS中几个常用s.
.ORACLE大數据類型的操作之一CLOB.
.oracle 使用杂记2.
.ExactPapers Oracle 1Z0-007 200.
.Oracle 9i 数据库设计指引全集(.
.如何在两个oracle服务器之间交换.
.linux学习笔记:进程及作业.
.新手问几个设置的问题。.
.Oracle复合数据类型(3).
.oracle8.1.7 在redhat7.2下的安装.
.RESETLOGS后没有备份情况下的数据.

PL/SQL中使用or展开进行sql调整

发表日期:2008-2-9



  问题:
  
  这样一条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是不需要排序的
上一篇:使用PL/SQL从数据库中读取BLOB对象 人气:801
下一篇:在PL/SQL中使DBMS_APPLICATION_INFO 人气:1022
浏览全部Oracle教程的内容 Dreamweaver插件下载 网页广告代码 祝你圣诞节快乐 2009年新年快乐