2022年SQL多表查询优化 .pdf
《2022年SQL多表查询优化 .pdf》由会员分享,可在线阅读,更多相关《2022年SQL多表查询优化 .pdf(7页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、SQL多表查询优化2009-12-07 11:00 这里提供的是执行性能的优化, 而不是后台数据库优化器资料: 参考数据库开发性能方面的各种问题, 收集了一些优化方案统计如下( 当然, 象索引等优化方案太过简单就不列入了, 嘿嘿): 执行路径 :ORACLE 的这个功能大大地提高了SQL的执行性能并节省了内存的使用: 我们发现 , 单表数据的统计比多表统计的速度完全是两个概念. 单表统计可能只要 0.02 秒, 但是 2 张表联合统计就可能要几十表了. 这是因为 ORACLE 只对简单的表提供 高速缓冲 (cache buffering) ,这个功能并不适用于 多表连接查询 . 数据库管理员必
2、须在init.ora中为这个区域设置合适的参数, 当这个内存区域越大, 就可以保留更多的语句 , 当然被共享的可能性也就越大了. 当你向 ORACLE 提交一个 SQL语句, ORACLE 会首先在这块内存中查找相同的语句. 这里需要注明的是 , ORACLE 对两者采取的是一种严格匹配, 要达成共享 ,SQL语句必须完全相同 (包括空格 , 换行等 ). 共享的语句必须满足三个条件: A. 字符级的比较 :当前被执行的语句和共享池中的语句必须完全相同. 例如: SELECT * FROM EMP; 和下列每一个都不同SELECT * from EMP; Select * From Emp;
3、SELECT * FROM EMP; B. 两个语句所指的对象必须完全相同:用户对象名如何访问Jack sal_limit private synonym Work_city public synonym Plant_detail public synonym Jill sal_limit private synonym Work_city public synonym Plant_detail table owner 考虑一下下列 SQL语句能否在这两个用户之间共享. SQL 能否共享原因select max(sal_cap) from sal_limit; 不能 每个用户都有一个priva
4、te synonym - sal_limit , 它们是不同的对象select count(*) from work_city where sdesc like NEW%; 能 两个用户访问相同的对象 public synonym - work_city select a.sdesc,b.location from work_city a , plant_detail b where 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 1 页,共 7 页 - - - - - - - -
5、- a.city_id = b.city_id 不能 用户 jack 通过 private synonym访问plant_detail 而 jill 是表的所有者 , 对象不同 . C. 两个 SQL语句中必须使用相同的名字的绑定变量(bind variables)例如:第一组的两个SQL语句是相同的 (可以共享 ), 而第二组中的两个语句是不同的( 即使在运行时 , 赋于不同的绑定变量相同的值) a. select pin , name from people where pin = :blk1.pin; select pin , name from people where pin = :
6、blk1.pin; b. select pin , name from people where pin = :blk1.ot_ind; select pin , name from people where pin = :blk1.ov_ind; 重点关注 1: 选择最有效率的表名顺序( 只在基于规则的优化器中有效) 重点关注ORACLE 的解析器按照从右到左的顺序处理FROM 子句中的表名 , 因此 FROM 子句中写在最后的表 ( 基础表 driving table)将被最先处理 . 在 FROM 子句中包含多个表的情况下 , 你必须选择记录条数最少的表作为基础表. 当 ORACLE 处
7、理多个表时, 会运用排序及合并的方式连接它们. 首先, 扫描第一个表 (FROM 子句中最后的那个表 )并对记录进行派序 , 然后扫描第二个表 (FROM 子句中最后第二个表 ),最后将所有从第二个表中检索出的记录与第一个表中合适记录进行合并. 例如: 表 TAB1 16,384 条记录表 TAB2 1 条记录选择 TAB2作为基础表 ( 最好的方法 ) select count(*) from tab1,tab2 执行时间 0.96 秒选择 TAB2作为基础表 ( 不佳的方法 ) select count(*) from tab2,tab1 执行时间 26.09 秒如果有 3个以上的表连接查
8、询 , 那就需要选择交叉表 (intersection table) 作为基础表 , 交叉表是指那个被其他表所引用的表. 例如: EMP表描述了 LOCATION 表和 CATEGORY表的交集 . SELECT * FROM LOCATION L , CATEGORY C, EMP E WHERE E.EMP_NO BETWEEN 1000 AND 2000 AND E.CAT_NO = C.CAT_NO AND E.LOCN = L.LOCN 将比下列 SQL更有效率SELECT * FROM EMP E , LOCATION L , CATEGORY C WHERE E.CAT_NO =
9、 C.CAT_NO AND E.LOCN = L.LOCN AND E.EMP_NO BETWEEN 1000 AND 2000 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 2 页,共 7 页 - - - - - - - - - 重点关注 2:WHERE 子句中的连接顺序重点关注ORACLE 采用自下而上的顺序解析WHERE 子句, 根据这个原理 , 表之间的连接必须写在其他 WHERE条件之前 , 那些可以过滤掉最大数量记录的条件必须写在WHERE子句的末尾 . 例如: ( 低
10、效, 执行时间 156.3 秒) SELECT FROM EMP E WHERE SAL ; 50000 AND JOB = MANAGER AND 25 (SELECT COUNT(*) FROM EMPWHERE MGR=E.EMPNO); ( 高效, 执行时间 10.6 秒) SELECT FROM EMP E WHERE 25 ; 50000 AND JOB = MANAGER; 重点关注 3:SELECT子句中避免使用 * 重点关注当你想在 SELECT 子句中列出所有的COLUMN时, 使用动态 SQL 列引用* 是一个方便的方法 . 不幸的是 , 这是一个非常低效的方法 . 实际
11、上 , ORACLE 在解析的过程中 , 会将* 依次转换成所有的列名, 这个工作是通过 查询数据字典完成的, 这意味着将耗费更多的时间. 7. 减少访问数据库的次数当执行每条 SQL语句时, ORACLE 在内部执行了许多工作 : 解析 SQL语句, 估算索引的利用率 , 绑定变量 , 读数据块等等 . 由此可见 , 减少访问数据库的次数 , 就能实际上减少 ORACLE 的工作量 . 例如, 以下有三种方法可以检索出雇员号等于0342 或 0291 的职员 . 方法 1 ( 最低效 ) SELECT EMP_NAME , SALARY , GRADE FROM EMP WHERE EMP_
12、NO = 342; SELECT EMP_NAME , SALARY , GRADE FROM EMP WHERE EMP_NO = 291; 方法 2 ( 次低效 ) DECLARE CURSOR C1 (E_NO NUMBER) IS SELECT EMP_NAME,SALARY,GRADE FROM EMP 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 3 页,共 7 页 - - - - - - - - - WHERE EMP_NO = E_NO; BEGIN OPEN C1
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 2022年SQL多表查询优化 2022 SQL 查询 优化
限制150内