It is because the statistics are *hard to get*.
Look at the query, assuming you would not be running part of the query to optimizer the query (because the optimizer doesn't), how many rows would you think would come out of it.
You know what the type is - 'INDEX'.
You know that gets one row from the type code table.
You know that you join to T by type_code.
That is all you know, how do you predict how many rows you'll get from that?
ops$tkyte%ORA11GR2> create table t1 as select * from all_objects;
Table created.
ops$tkyte%ORA11GR2> create table t2 as select * from all_users;
Table created.
ops$tkyte%ORA11GR2>
ops$tkyte%ORA11GR2> alter table t2 add constraint t2_pk primary key(username);
Table altered.
ops$tkyte%ORA11GR2> create bitmap index t1_t2_bidx
2 on t1( t2.user_id )
3 from t1, t2
4 where t1.owner = t2.username;
Index created.
ops$tkyte%ORA11GR2>
ops$tkyte%ORA11GR2> exec dbms_stats.gather_table_stats( user, 'T1' );
PL/SQL procedure successfully completed.
ops$tkyte%ORA11GR2> exec dbms_stats.gather_table_stats( user, 'T2' );
PL/SQL procedure successfully completed.
ops$tkyte%ORA11GR2>
ops$tkyte%ORA11GR2> set autotrace traceonly
ops$tkyte%ORA11GR2> select count(*)
2 from t1, t2
3 where t1.owner = t2.username
4 and t2.user_id = 42;
Execution Plan
----------------------------------------------------------
Plan hash value: 758379198
------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | 6 | 1 (0)| 00:00:01 |
| 1 | SORT AGGREGATE | | 1 | 6 | | |
| 2 | BITMAP CONVERSION COUNT | | 2187 | 13122 | 1 (0)| 00:00:01 |
|* 3 | BITMAP INDEX SINGLE VALUE| T1_T2_BIDX | | | | |
------------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
3 - access("T1"."SYS_NC00016$"=42)
Statistics
----------------------------------------------------------
1 recursive calls
0 db block gets
2 consistent gets
0 physical reads
0 redo size
423 bytes sent via SQL*Net to client
420 bytes received via SQL*Net from client
2 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
1 rows processed
ops$tkyte%ORA11GR2>
ops$tkyte%ORA11GR2> select t1.* from t1, t2 where t1.owner = t2.username and t2.user_id = 42;
308 rows selected.
Execution Plan
----------------------------------------------------------
Plan hash value: 3169558978
-------------------------------------------------------------------------------------------
| Id | Operation | Name | Rows | Bytes | Cost (%CPU)| Time |
-------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 2187 | 207K| 231 (0)| 00:00:03 |
| 1 | TABLE ACCESS BY INDEX ROWID | T1 | 2187 | 207K| 231 (0)| 00:00:03 |
| 2 | BITMAP CONVERSION TO ROWIDS| | | | | |
|* 3 | BITMAP INDEX SINGLE VALUE | T1_T2_BIDX | | | | |
-------------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
3 - access("T1"."SYS_NC00016$"=42)
Statistics
----------------------------------------------------------
1 recursive calls
0 db block gets
31 consistent gets
0 physical reads
0 redo size
33874 bytes sent via SQL*Net to client
640 bytes received via SQL*Net from client
22 SQL*Net roundtrips to/from client
0 sorts (memory)
0 sorts (disk)
308 rows processed
ops$tkyte%ORA11GR2> set autotrace off