Performance Improvement – Index scan with NULL condition
What will be our first step when we start tuning the queries?
The first step is creating the index. But simply creating the index does not improve the performance in all scenarios.
The below scripts are proves that.
SQL> create table t
2 as
3 select object_name name, a.*
4 from all_objects a;
Table created.
SQL> alter table t modify name null;
Table altered.
SQL> create index t_idx on t(name);
Index created.
SQL> exec dbms_stats.gather_table_stats( user, 'T' );
PL/SQL procedure successfully completed.
SQL> variable b1 varchar2(30)
SQL> exec :b1 := 'T'
PL/SQL procedure successfully completed.
SQL> set autotrace traceonly explain
SQL> delete from t
2 where name = :b1 or name is null;
2 rows deleted.
Execution Plan
----------------------------------------------------------
0 DELETE STATEMENT Optimizer=CHOOSE (Cost=49 Card=2 Bytes=48)
1 0 DELETE OF 'T'
2 1 TABLE ACCESS (FULL) OF 'T' (Cost=49 Card=2 Bytes=48)
The NAME field is having the index but still it is going for the FULL table scan. the index (t_idx) is not getting used here.
The reason is because NAME is nullable, the index on only on name and entirely null keys are NOT entered into b*tree indexes.
SQL> rollback;
Rollback complete.
SQL> drop index t_idx;
Index dropped.
SQL> create index t_idx on t(name,0);
Index created.
SQL> exec dbms_stats.gather_table_stats( user, 'T' );
PL/SQL procedure successfully completed.
SQL> delete from t
2 where name = :b1 or name is null
3 ;
2 rows deleted.
Execution Plan
----------------------------------------------------------
0 DELETE STATEMENT Optimizer=CHOOSE (Cost=26 Card=2 Bytes=48)
1 0 DELETE OF 'T'
2 1 INDEX (FULL SCAN) OF 'T_IDX' (NON-UNIQUE) (Cost=26 Card=
2 Bytes=48)
The above method is going for the index scan and avoids the table scan.
What will be our first step when we start tuning the queries?
The first step is creating the index. But simply creating the index does not improve the performance in all scenarios.
The below scripts are proves that.
SQL> create table t
2 as
3 select object_name name, a.*
4 from all_objects a;
Table created.
SQL> alter table t modify name null;
Table altered.
SQL> create index t_idx on t(name);
Index created.
SQL> exec dbms_stats.gather_table_stats( user, 'T' );
PL/SQL procedure successfully completed.
SQL> variable b1 varchar2(30)
SQL> exec :b1 := 'T'
PL/SQL procedure successfully completed.
SQL> set autotrace traceonly explain
SQL> delete from t
2 where name = :b1 or name is null;
2 rows deleted.
Execution Plan
----------------------------------------------------------
0 DELETE STATEMENT Optimizer=CHOOSE (Cost=49 Card=2 Bytes=48)
1 0 DELETE OF 'T'
2 1 TABLE ACCESS (FULL) OF 'T' (Cost=49 Card=2 Bytes=48)
The NAME field is having the index but still it is going for the FULL table scan. the index (t_idx) is not getting used here.
The reason is because NAME is nullable, the index on only on name and entirely null keys are NOT entered into b*tree indexes.
SQL> rollback;
Rollback complete.
SQL> drop index t_idx;
Index dropped.
SQL> create index t_idx on t(name,0);
Index created.
SQL> exec dbms_stats.gather_table_stats( user, 'T' );
PL/SQL procedure successfully completed.
SQL> delete from t
2 where name = :b1 or name is null
3 ;
2 rows deleted.
Execution Plan
----------------------------------------------------------
0 DELETE STATEMENT Optimizer=CHOOSE (Cost=26 Card=2 Bytes=48)
1 0 DELETE OF 'T'
2 1 INDEX (FULL SCAN) OF 'T_IDX' (NON-UNIQUE) (Cost=26 Card=
2 Bytes=48)
The above method is going for the index scan and avoids the table scan.