优化select count(*) from t1

分享下如何优化一条简单的SQL语句,数据库优化无止境,学习数据库优化,我们先从一条最简单的SQL语句开始。
select count(*) from t1;
这条语句虽然比较简单,但很有玄机!对这句话运行的理解,反映了你对数据库的理解深度!
好!我们开始我们的实验。
建立实验的使用的大表他t1

SQL> conn scott/tiger
已连接。
SQL> drop table t1 purge;

表已删除。

SQL> create table t1 as select * from emp where 0=9;

表已创建。

SQL> insert into t1 select * from emp;

已创建14行。

SQL>  insert into t1 select * from t1;

已创建14行。

SQL> /

已创建28行。

SQL> /

已创建56行。

SQL> /

已创建112行。

SQL> /

已创建224行。

SQL> /

已创建448行。

SQL> /

已创建896行。

SQL> /

已创建1792行。

SQL> /

已创建3584行。

SQL> /

已创建7168行。

SQL> /

已创建14336行。

SQL> /

已创建28672行。

SQL> /

已创建57344行。

SQL> commit;

提交完成。

收集统计信息
SQL> execute dbms_stats.gather_table_stats('SCOTT','T1');

PL/SQL 过程已成功完成。

SQL> SET AUTOT TRACE EXP
SQL> SELECT COUNT(*) FROM T1;

执行计划
--------------------------------------------------                               
| Id  | Operation          | Name | Rows  | Cost (%CPU)| Time     |                                
-------------------------------------------------------------------                                
|   0 | SELECT STATEMENT   |      |     1 |   124 (4)| 00:00:02 |                                
|   1 |  SORT AGGREGATE    |      |     1 |          |          |                                
|   2 |   TABLE ACCESS FULL| T1   |   116K|   124 (4)| 00:00:02 |                                
-----------------------------------------------------
代价为124,运行的计划为全表扫描。 
那么我们删除表中所有的数据,是不是代价就能少了那?按照我们的正常思维是这样的,但是oracle数据库
没有这么做,没有那么智能,毕竟它只是一个程序。
我们删除T1的所有数据,我们来看看。            
SQL> DELETE T1 WHERE DEPTNO=10;

已删除24576行。

SQL> COMMIT;

提交完成。

SQL> execute dbms_stats.gather_table_stats('SCOTT','T1');

PL/SQL 过程已成功完成。

SQL> SELECT COUNT(*) FROM T1;

执行计划
-----------------------------------------------------
| Id  | Operation          | Name | Rows  | Cost (%CPU)| Time     |                                
-------------------------------------------------------------------                                
|   0 | SELECT STATEMENT   |      |     1 |   123 (3)| 00:00:02 |                                
|   1 |  SORT AGGREGATE    |      |     1 |          |          |                                
|   2 |   TABLE ACCESS FULL| T1   | 90286 |  123  (3)| 00:00:02 |                                
-----------------------------------------------------
我们看到了,代价为123几乎没有变化。
第一种优化的手段:
--1.降低高水位
SQL> alter table t1 move tablespace users;

表已更改。

SQL> execute dbms_stats.gather_table_stats('SCOTT','T1');

PL/SQL 过程已成功完成。

SQL> SELECT COUNT(*) FROM T1;

执行计划
-----------------------------------------------------
| Id  | Operation          | Name | Rows  |Cost (%CPU)| Time   |                                
-------------------------------------------------------------------                                
|   0 | SELECT STATEMENT   |      |     1 |   102 (3)| 00:00:02 |                                
|   1 |  SORT AGGREGATE    |      |     1 |          |          |                                
|   2 |   TABLE ACCESS FULL| T1   | 90667 |   102 (3)| 00:00:02 |                                
-----------------------------------------------------                         
代价为102,降低了,但是不多。

第二个手段是:
2.修改pctfree
SQL> alter table t1 pctfree 0;

表已更改。

SQL> alter table t1 move tablespace users;

表已更改。

SQL> execute dbms_stats.gather_table_stats('SCOTT','T1');

PL/SQL 过程已成功完成。

SQL> SELECT COUNT(*) FROM T1;

执行计划
----------------------------------------------------------                                         
Plan hash value: 3724264953                                                                        
                                                                                                   
-------------------------------------------------------------------                                
| Id  | Operation          | Name | Rows  | Cost (%CPU)| Time     |                                
-------------------------------------------------------------------                                
|   0 | SELECT STATEMENT   |      |     1 |    92   (4)| 00:00:02 |                                
|   1 |  SORT AGGREGATE    |      |     1 |            |          |                                
|   2 |   TABLE ACCESS FULL| T1   | 91791 |    92   (4)| 00:00:02 |                                
-------------------------------------------------------------------                                
代价为92,降低了10%
第三种手段是:

--3.参数db_file_multiblock_read_count=64
该参数是一次性读的数据库块数。一次性读的块数越多。数据库的性能就越高。
第四种手段是:
--4.建立b*tree类型的索引
SQL> create index i1 on t1(empno);

索引已创建。

SQL> execute dbms_stats.gather_index_stats('SCOTT','I1');

PL/SQL 过程已成功完成。

SQL> SELECT COUNT(*) FROM T1;

执行计划
----------------------------------------------------------                                         
Plan hash value: 3724264953                                                                        
                                                                                                   
-------------------------------------------------------------------                                
| Id  | Operation          | Name | Rows  | Cost (%CPU)| Time     |                                
-------------------------------------------------------------------                                
|   0 | SELECT STATEMENT   |      |     1 |    92   (4)| 00:00:02 |                                
|   1 |  SORT AGGREGATE    |      |     1 |            |          |                                
|   2 |   TABLE ACCESS FULL| T1   | 91791 |    92   (4)| 00:00:02 |                                
-------------------------------------------------------------------                                
为什么没有使用我们建立的索引,因为null不进入普通的索引!

SQL> alter table t1 modify(empno not null);

表已更改。

SQL> SELECT COUNT(*) FROM T1;

执行计划
----------------------------------------------------------                                         
Plan hash value: 129980005                                                                         
                                                                                                   
----------------------------------------------------------------------                             
| Id  | Operation             | Name | Rows  | Cost (%CPU)| Time     |                             
----------------------------------------------------------------------                             
|   0 | SELECT STATEMENT      |      |     1 |    36   (6)| 00:00:01 |                             
|   1 |  SORT AGGREGATE       |      |     1 |            |          |                             
|   2 |   INDEX FAST FULL SCAN| I1   | 91791 |    36   (6)| 00:00:01 |                             
----------------------------------------------------------------------                             
代价为36,我们的索引起到了很大的作用!

第五种手段是:

SQL> --5.使用并行查询的特性
                               
强制全表扫描,屏蔽索引

SQL> select /*+ full(t1) parallel(t1 2) */ COUNT(*) FROM T1;

执行计划
----------------------------------------------------------------------------------------
| Id  | Operation    | Name     | Rows  | Cost (%CPU)| Time     |    TQ  |IN-OUT| PQ Distrib |     
----------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |    |     1 |    51   (4)| 00:00:01 |    |      |    | 
|   1 |  SORT AGGREGATE        |          |     1 |    |    |        |      |    |   
|   2 |   PX COORDINATOR       |   |       |            |          |        |    |    |           
|   3 |    PX SEND QC (RANDOM) | :TQ10000 |     1 |    |   |  Q1,00 | P->S | QC(RAND)  |        
|   4 |     SORT AGGREGATE     |          |     1 |    |  |  Q1,00 | PCWP |  |                  
|   5 |      PX BLOCK ITERATOR |          | 91791 |    51   (4)| 00:00:01 |  Q1,00 | PCWC|  |   
|   6 |       TABLE ACCESS FULL| T1       | 91791 |    51   (4)| 00:00:01 |  Q1,00 | PCWP |  |    
-----------------------------------------------------------------------------------------------
并行度越高,代价越低

SQL> alter table t1 parallel 4;

表已更改。
也可以通过使用表的属性来定义并行度,但是影响比较大,不如语句级别限制并行!

SQL> select count(*) from t1;

执行计划
-----------------------------------------------------------------------------------------
| Id  | Operation      | Name     | Rows  | Cost (%CPU)| Time   |    TQ  |IN-OUT| PQDistrib |    
---------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |      |     1 |    25   (0)| 00:00:01 |    |    |    |        
|   1 |  SORT AGGREGATE        |       |     1 |      |          |        |      |     |        
|   2 |   PX COORDINATOR       |          |       |        |      |        |      |    |        
|   3 |    PX SEND QC (RANDOM) | :TQ10000 |     1 |       |   |  Q1,00 | P->S | QC(RAND)  |     
|   4 |     SORT AGGREGATE     |          |     1 |      |      |  Q1,00 | PCWP |    |         
|   5 |      PX BLOCK ITERATOR |          | 91791 |    25   (0)| 00:00:01 |  Q1,00 | PCWC |   |  
|   6 |       TABLE ACCESS FULL| T1       | 91791 |    25   (0)| 00:00:01 |  Q1,00 | PCWP |   | 
---------------------------------------------------------------------------------------------
代价为25,代价比两个的又少一半!
第六种手段是:

SQL> --6.建立位图索引来避免全表扫描
SQL> create bitmap index i2 on t1(deptno);

索引已创建。

SQL> execute dbms_stats.gather_index_stats('SCOTT','I2');

PL/SQL 过程已成功完成。

SQL> select count(*) from t1;

执行计划
----------------------------------------------------------                                         
Plan hash value: 3738977131                                                                        
                                                                                                   
------------------------------------------------------------------------------                     
| Id  | Operation                     | Name | Rows  | Cost (%CPU)| Time     |                     
------------------------------------------------------------------------------                     
|   0 | SELECT STATEMENT              |      |     1 |     4   (0)| 00:00:01 |                     
|   1 |  SORT AGGREGATE               |      |     1 |            |          |                     
|   2 |   BITMAP CONVERSION COUNT     |      | 91791 |     4   (0)| 00:00:01 |                     
|   3 |    BITMAP INDEX FAST FULL SCAN| I2   |       |            |          |                     
------------------------------------------------------------------------------                     

SQL> alter index i2 parallel 4;

索引已更改。

SQL> select count(*) from t1;
执行计划
----------------------------------------------------------------------------------------
| Id  | Operation       | Name   | Rows  | Cost (%CPU)| Time   |   TQ  |IN-OUT| PQ Distrib |    
----------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT     |      |     1 |     2   (0)| 00:00:01 |      | |       |          
|   1 |  SORT AGGREGATE   |   |     1 |            |          |        |  |      |               
|   2 |   PX COORDINATOR   |      |       |       |          |        |  |    |               
|   3 |    PX SEND QC (RANDOM) | :TQ10000 |     1 |    |          |  Q1,00 | P->S | QC (RAND) |   
|   4 |     SORT AGGREGATE  |    |     1 |   |      |  Q1,00 | PCWP |        |        
|   5 |      PX BLOCK ITERATOR |  | 91791 |     2   (0)| 00:00:01 |  Q1,00 |PCWC |     |        
|   6 |       BITMAP CONVERSION COUNT  |   | 91791 |  2   (0)| 00:00:01 |  Q1,00 |PCWP |    |  
|   7 |        BITMAP INDEX FAST FULL SCAN| I2    |   |     |     |  Q1,00 | PCWP |    |        
--------------------------------------------------------------------------------------------
代价为2,原来为124,优化无止境呀!
只有你把握原理,一切尽在掌握!

要去开早会了,今天就到这里吧,希望大家能学到一点有用的知识。在工作中能用到。



pxboracle@live.com
2014.08.12 08:24
share you knowledge with the world. 


来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/12798004/viewspace-1249555/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/12798004/viewspace-1249555/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值