PL/SQL - Optional conditions in where-clause - without dynamic sql?

后端 未结 5 1821
隐瞒了意图╮
隐瞒了意图╮ 2021-02-08 13:51

I have a query where not all conditions are necessary. Here\'s an example of what it looks like when all conditions are used:

select num
from (select distinct q.         


        
5条回答
  •  孤街浪徒
    2021-02-08 14:17

    While you could do this...

    select num
    from (select distinct q.num
           from cqqv q
           where 1=1
                 and (:bcode is null or q.bcode = :bcode)
                 and (:lb is null or q.lb = :lb)
                 and (:type is null or q.type = :type)
                 and (:edate is null or q.edate > :edate - 30)
           order by dbms_random.value()) subq
    where rownum <= :numrows
    

    ... the performance using dynamic SQL will usually be better, as it will generate a more targeted query plan. In the above query, Oracle cannot tell whether to use an index on bcode or lb or type or edate, and will probably perform a full table scan every time.

    Of course, you must use bind variables in your dynamic query, not concatenate the literal values into the string, otherwise performance (and scalability, and security) will be very bad.

    To be clear, the dynamic version I have in mind would work like this:

    declare
        rc sys_refcursor;
        q long;
    begin
        q := 'select num
        from (select distinct q.num
               from cqqv q
               where 1=1';
    
        if p_bcode is not null then
            q := q || 'and q.bcode = :bcode';
        else
            q := q || 'and (1=1 or :bcode is null)';
        end if;
    
        if p_lb is not null then
            q := q || 'and q.lb = :lb';
        else
            q := q || 'and (1=1 or :lb is null)';
        end if;
    
        if p_type is not null then
            q := q || 'and q.type = :type';
        else
            q := q || 'and (1=1 or :type is null)';
        end if;
    
        if p_edate is not null then
            q := q || 'and q.edate = :edate';
        else
            q := q || 'and (1=1 or :edate is null)';
        end if;
    
        q := q || ' order by dbms_random.value()) subq
        where rownum <= :numrows';
    
        open rc for q using p_bcode, p_lb, p_type, p_edate, p_numrows;
        return rc;
    end;
    

    This means that the result query will be "sargable" (a new word to me I must admit!) since the resulting query run will be (for example):

    select num
    from (select distinct q.num
           from cqqv q
           where 1=1
                 and q.bcode = :bcode
                 and q.lb = :lb
                 and (1=1 or :type is null)
                 and (1=1 or :edate is null)
           order by dbms_random.value()) subq
    where rownum <= :numrows
    

    However, I accept that this could require up to 16 hard parses in this example. The "and :bv is null" clauses are required when using native dynamic SQL, but could be avoided by using DBMS_SQL.

    Note: the use of (1=1 or :bindvar is null) when the bind variable is null was suggested in a comment by Michal Pravda, as it allows the optimizer to eliminate the clause.

提交回复
热议问题