How to handle optional parameters in SQL query?

后端 未结 3 796
醉梦人生
醉梦人生 2020-12-03 12:50

Say I have a sample table:

 id_pk  value
------------
 1       a
 2       b
 3       c

And I have a sample PL/SQL block, which has a quer

3条回答
  •  悲&欢浪女
    2020-12-03 13:22

    The NVL approach will usually work fine. The optimizer recognizes this pattern and will build a dynamic plan. The plan uses an index for a single value and a full table scan for a NULL.

    Sample table and data

    drop table myTable;
    create table myTable(
        id_pk number,
        value varchar2(100),
        constraint myTable_pk primary key (id_pk)
    );
    
    insert into myTable select level, level from dual connect by level <= 100000;
    commit;
    

    Execute with different predicates

    --Execute predicates that return one row if the ID is set, or all rows if ID is null. 
    declare
        type t_table is table of myTable%rowtype;
        n_RequiredId myTable.id_pk%type := 1;  
        t_Output t_table := t_table();
    begin
        select /*+ SO_QUERY_1 */ m.id_pk, m.value
        bulk collect into t_Output
        from myTable m
        where m.id_pk = nvl(n_RequiredId, m.id_pk);
    
        select /*+ SO_QUERY_2 */ m.id_pk, m.value
        bulk collect into t_Output
        from myTable m
        where m.id_pk = COALESCE(n_RequiredId, m.id_pk);
    
        select /*+ SO_QUERY_3 */ m.id_pk, m.value
        bulk collect into t_Output
        from myTable m
        where (n_RequiredId IS NULL OR m.id_pk = n_RequiredId);
    end;
    /
    

    Get execution plans

    select sql_id, child_number
    from gv$sql
    where lower(sql_text) like '%so_query_%'
        and sql_text not like '%QUINE%'
        and sql_text not like 'declare%';
    
    select * from table(dbms_xplan.display_cursor(sql_id => '76ucq3bkgt0qa', cursor_child_no => 1, format => 'basic'));
    select * from table(dbms_xplan.display_cursor(sql_id => '4vxf8yy5xd6qv', cursor_child_no => 1, format => 'basic'));
    select * from table(dbms_xplan.display_cursor(sql_id => '457ypz0jpk3np', cursor_child_no => 1, format => 'basic'));
    

    Bad plans for COALESCE and IS NULL OR

    EXPLAINED SQL STATEMENT:
    ------------------------
    SELECT /*+ SO_QUERY_2 */ M.ID_PK, M.VALUE FROM MYTABLE M WHERE M.ID_PK 
    = COALESCE(:B1 , M.ID_PK)
    
    Plan hash value: 1229213413
    
    -------------------------------------
    | Id  | Operation         | Name    |
    -------------------------------------
    |   0 | SELECT STATEMENT  |         |
    |   1 |  TABLE ACCESS FULL| MYTABLE |
    -------------------------------------
    
    
    EXPLAINED SQL STATEMENT:
    ------------------------
    SELECT /*+ SO_QUERY_3 */ M.ID_PK, M.VALUE FROM MYTABLE M WHERE (:B1 IS 
    NULL OR M.ID_PK = :B1 )
    
    Plan hash value: 1229213413
    
    -------------------------------------
    | Id  | Operation         | Name    |
    -------------------------------------
    |   0 | SELECT STATEMENT  |         |
    |   1 |  TABLE ACCESS FULL| MYTABLE |
    -------------------------------------
    

    Good plan for NVL

    The FILTER operations allow the optimizer to choose a different plan at run time, depending on the input values.

    EXPLAINED SQL STATEMENT:
    ------------------------
    SELECT /*+ SO_QUERY_1 */ M.ID_PK, M.VALUE FROM MYTABLE M WHERE M.ID_PK 
    = NVL(:B1 , M.ID_PK)
    
    Plan hash value: 730481884
    
    ----------------------------------------------------
    | Id  | Operation                     | Name       |
    ----------------------------------------------------
    |   0 | SELECT STATEMENT              |            |
    |   1 |  CONCATENATION                |            |
    |   2 |   FILTER                      |            |
    |   3 |    TABLE ACCESS FULL          | MYTABLE    |
    |   4 |   FILTER                      |            |
    |   5 |    TABLE ACCESS BY INDEX ROWID| MYTABLE    |
    |   6 |     INDEX UNIQUE SCAN         | MYTABLE_PK |
    ----------------------------------------------------
    

    Warnings

    FILTER operations and this NVL trick are not well documented. I'm not sure what version introduced these features but it works with 11g. I've had problems getting the FILTER to work correctly with some complicated queries, but for simple queries like these it is reliable.

提交回复
热议问题