这是个终极问题,因为优化本身的复杂性实在是难以总结的,很多时候优化的方法并不是用到了什么高深莫测的技术,而只是一个思想意识层面的差异,而这些都很可能连带导致性能表现上的巨大差异。

所以有时候我们应该先搞清楚需求到底是什么,SQL本身是否合理,这些思考很可能会使优化工作事半功倍。而本文是假设SQL本身合理,从Oracle提供给我们的一些技术手段来简单介绍下Oracle数据库,该如何使用一些现有的技术来优化一个SQL执行的性能。

  1. 确定需要优化的SQL文本及当前SQL执行计划
  2. 确定SQL涉及的所有表及其索引的相关信息
  3. 运行SQL Tuning Advisor 得到调整建议供优化参考
  4. 收集表信息
  5. 收集索引信息
  6. SQL Profile
  7. 物化视图

1. 确定需要优化的SQL文本及当前SQL执行计划

优化之前先确定好需要优化的SQL文本以及当前SQL的执行计划是什么样,注意PL/SQL Developer这类工具F5看到的执行计划很可能并不准确。
相关内容参考:
- [SQL Tuning 基础概述01 - Autotrace的设定](http://www.cnblogs.com/jyzhao/p/3849893.html)
- [SQL Tuning 基础概述02 - Explain plan的使用](http://www.cnblogs.com/jyzhao/p/3850022.html)
- [SQL Tuning 基础概述03 - 使用sql_trace和10046事件跟踪执行计划](http://www.cnblogs.com/jyzhao/p/3850047.html)

2. 确定SQL涉及的所有表及其索引的相关信息

确定查询涉及到的所有表及其索引的相关基础信息。比如:
> 各表的数据量
> 表和索引类型
> 表分区信息,每个分区的数据量
> 索引字段
> 索引分区信息
> 表关联方式
> 结果集的数量

确定相关信息,以T2表为例:

--普通表/分区表信息
select * from dba_tables where table_name = 'T2';
select * from dba_part_tables where table_name = 'T2';
--普通表/分区表的每个分区大约__G大小
select (t.bytes/1024/1024) "MB", t.* from dba_segments t where segment_name = 'T2'; --表数据量信息
--普通表的数据量
select count(1) from T2; --____数据左右
--分区表的某个分区数据量
select count(*) from T2 partition(P20160101); --____数据左右
select count(*) from T2 partition(P20160102); --表索引信息
--普通表索引及各个索引的索引列
select * from dba_indexes where table_name = 'T2';
select * from dba_ind_columns where index_name in (select index_name from dba_indexes where table_name = 'T2')order by index_name, column_position;
--分区表索引及各个索引的索引列
select * from dba_part_indexes where table_name = 'T2';
select * from dba_ind_columns where index_name in (select index_name from dba_part_indexes where table_name = 'T2') order by index_name, column_position;
--索引段大小信息
--select (t.bytes/1024/1024) "MB", t.* from dba_segments t where segment_name in (select index_name from dba_part_indexes where table_name = 'T2') order by segment_name, partition_name;

相关内容参考:

3. 运行SQL Tuning Advisor 得到调整建议供优化参考

运行SQL Tuning Advisor 得到调整建议供优化参考, SQL Tuning Advisor得到的优化建议仅供参考,具体如何做还需要结合业务实际情况。

相关内容参考:

4. 收集表信息

例如收集ZJY用户下T2表的统计信息。(T2是range分区表,按天分区,每天数据量大概80w,存放半年)
```
SQL> execute dbms_stats.gather_table_stats(ownname => 'ZJY', tabname => 'T2', estimate_percent => DBMS_STATS.AUTO_SAMPLE_SIZE, method_opt => 'FOR ALL COLUMNS SIZE AUTO', cascade => TRUE, degree => 16);

PL/SQL procedure successfully completed

Executed in 5896.641 seconds

统计信息加锁/解锁

--锁住表的统计信息

exec dbms_stats.lock_table_stats('ZJY','T2');

--解锁表的统计信息

exec dbms_stats.unlock_table_stats('ZJY','T2');

相关内容参考:
- [【转载】dbms_stats 导入导出表统计信息](http://www.cnblogs.com/jyzhao/articles/4746972.html) <h1 id="5">5. 收集索引信息</h1>
例如只收集ZJY用户下T2表的索引IDX_T2_1统计信息。(IDX_T2_1是分区索引,包含4个字段)

SQL> execute dbms_stats.gather_index_stats(ownname => 'ZJY', indname => 'IDX_T2_1', estimate_percent => DBMS_STATS.AUTO_SAMPLE_SIZE, degree => 8);

PL/SQL procedure successfully completed

Executed in 44.312 seconds

有时还很可能需要在业务闲时在线创建新的索引

--不记录日志在线并行创建单列索引IDX_T2_2(并行度视生产环境当前的CPU资源使用情况来确定合理的值)

create index IDX_T2_2 on T2(start_time) tablespace DBS_I_JINGYU nologging parallel 12 online;

alter index IDX_T2_2 noparallel;

alter index IDX_T2_2 logging;

<h1 id="6">6. SQL Profile</h1>
SQL Profile是10g中的新特性,作为自动SQL调整过程的一部分。SQL Profile是一个对象,它包含了可以帮助查询优化器为一个特定的SQL语句找到高效执行计划的信息。这些信息包括执行环境、对象统计和对查询优化器所做评估的修正信息。它的最大优点之一就是在不修改SQL语句和会话执行环境的情况下影响查询优化器的决定。SQL Profile中包含的并非单个执行计划的信息,SQL Profile不会固定一个SQL语句的执行计划。当表的数据增长或者索引创建、删除,使用同一个SQL Profile的执行计划可能会改变,而存储在SQL Profile中的信息会继续起作用。所以,经过一段很长的时间之后,它的信息有可能会过时,需要重新生成。 相关内容参考:
- [【转载】sql_profile的使用](http://www.cnblogs.com/jyzhao/articles/3766455.html)
- [【转载】sql_profile快速绑定脚本](http://www.cnblogs.com/jyzhao/articles/3766713.html)
- [使用COE脚本绑定SQL Profile](https://www.cnblogs.com/jyzhao/p/9256293.html) <h1 id="7">7. 物化视图</h1>
Oracle的物化视图可以用于预先计算并保存(表连接或聚集等耗时较多的操作的)结果,所以合理使用物化视图,会在执行查询时避免进行这些耗时的操作,从而快速的得到结果。 相关内容参考:
- [【转载】物化视图详解](http://www.cnblogs.com/jyzhao/articles/5124141.html)

Oracle数据库该如何着手优化一个SQL的更多相关文章

  1. 使用oracle数据库,多用户同时对一个表进行增加,删除,修改,查看等操作,会不会有影响?

    使用oracle数据库,多用户同时对一个表进行增加,删除,修改,查看等操作,会不会有影响? 1.问题:各操作间或者性能上会不会有影响? 如果有该如何解决? 多用户操作的影响主要是回锁定记录,oracl ...

  2. Oracle数据库软件标准版的一个限制:仅仅能用一个rman channel

    Oracle数据库软件标准版的一个限制:仅仅能用一个rman channel Restrictions in "Standard Edition" Rman channel all ...

  3. ORACLE 数据库的级联查询 一句sql搞定(部门多级)

    在ORACLE 数据库中有一种方法可以实现级联查询   select  *                //要查询的字段 from table              //具有子接点ID与父接点I ...

  4. Oracle数据库在给表添加字段的sql中用comment报错

    原因:不同于mysql,Oracle数据库在添加表字段时不能直接用comment,而是单独写一个sql语句,如下: alter table SYS_USER add SENDMSG_LASTTIME ...

  5. oracle数据库、客户端安装以及ps/sql连接和导入表实例

    从下面的网址下载http://www.oracle.com/technetwork/database/enterprise-edition/downloads/112010-win32soft-098 ...

  6. 第一安装oracle数据库后,需要创建一个用户,给用户解锁并赋予权限

    1.第一次安装oracle数据库应该做的事情. 注: 1.安装oracle后需要创建用户,连接数据库,(注意数据库名,还有好像后面的 ":"也有影响) 2.解锁用户, 3.授予新登 ...

  7. 吴裕雄--天生自然ORACLE数据库学习笔记:优化SQL语句

    create or replace procedure trun_table(table_deleted in varchar2) as --创建一个存储过程,传入一个表示表名称的参数,实现清空指定的 ...

  8. Oracle数据库概念和一些基本的SQL语句

    1.数据 定义:描述事物的符号.例如:文本.音频.视频都是数据. 2.数据库 存放数据的仓库,存放在计算机中,按照一定格式存放,可以为用户共享. 3.数据库的发展阶段 1.网状数据库 2.层次数据库 ...

  9. C#数据库——用多线程来组合一个SQL语句

    StringBuffer sql = new StringBuffer();或(StringBuilder sql = new StringBuilder ()) sql.Append("s ...

随机推荐

  1. Linux CentOS 配置Tomcat环境

    一.下载Tomcat 下载Tomcat方式也有两种,可以参考我的前一篇博文Linux CentOS配置JDK环境,这边就不再赘述. 二.在Linux处理Tomcat包 1.创建tomcat文件夹 mk ...

  2. 了解PHP中的Array数组和foreach

    1. 了解数组 PHP 中的数组实际上是一个有序映射.映射是一种把 values 关联到 keys 的类型.详细的解释可参见:PHP.net中的Array数组    . 2.例子:一般的数组 这里,我 ...

  3. 快速搭建springmvc+spring data jpa工程

    一.前言 这里简单讲述一下如何快速使用springmvc和spring data jpa搭建后台开发工程,并提供了一个简单的demo作为参考. 二.创建maven工程 http://www.cnblo ...

  4. 高频交易算法研发心得--MACD指标算法及应用

    凤鸾宝帐景非常,尽是泥金巧样妆. 曲曲远山飞翠色:翩翩舞袖映霞裳. 梨花带雨争娇艳:芍药笼烟骋媚妆. 但得妖娆能举动,取回长乐侍君王. [摘自<封神演义>纣王在女娲宫上香时题的诗] 一首定 ...

  5. AFNetworking 3.0 源码解读(十一)之 UIButton/UIProgressView/UIWebView + AFNetworking

    AFNetworking的源码解读马上就结束了,这一篇应该算是倒数第二篇,下一篇会是对AFNetworking中的技术点进行总结. 前言 上一篇我们总结了 UIActivityIndicatorVie ...

  6. canvas与html5实现视频截图功能

    这段时间一直在研究canvas,突发奇想想做一个可以截屏视频的功能,然后把图片拉去做表情包,哈哈哈哈哈哈~~ 制作方法: 1.在页面中加载视频 在使用canvas制作这个截图功能时,首先必须保证页面上 ...

  7. css3制作旋转动画

    现在的css3真是强大,之前很多动画都是用jq来实现,但是css3制作的动画要比jq实现起来简单很多,今天呢,我自己也写了一个css旋转动画和大家分享.效果如下面的图片 思路:1.制作之前呢,我们先来 ...

  8. Linux实战教学笔记05:远程SSH连接服务与基本排错(新手扫盲篇)

    第五节 远程SSH连接服务与基本排错 标签(空格分隔):Linux实战教学笔记-陈思齐 第1章 远程连接LInux系统管理 1.1 为什么要远程连接Linux系统 在实际的工作场景中,虚拟机界面或物理 ...

  9. 机器指令翻译成 JavaScript —— 终极目标

    上一篇,我们顺利将 6502 指令翻译成 C 代码,并演示了一个案例. 现在,我们来完成最后的目标 -- 转换成 JavaScript. 中间码输出 我们之所以选择 C,就是为了使用 LLVM.现在来 ...

  10. ES6 箭头函数中的 this?你可能想多了(翻译)

    箭头函数=>无疑是ES6中最受关注的一个新特性了,通过它可以简写 function 函数表达式,你也可以在各种提及箭头函数的地方看到这样的观点——“=> 就是一个新的 function”. ...