| From: | Merlin Moncure <mmoncure(at)gmail(dot)com> | 
|---|---|
| To: | Linos <info(at)linos(dot)es> | 
| Cc: | Tom Lane <tgl(at)sss(dot)pgh(dot)pa(dot)us>, pgsql-performance(at)postgresql(dot)org | 
| Subject: | Re: CTE vs Subquery | 
| Date: | 2011-10-25 17:11:23 | 
| Message-ID: | CAHyXU0wTCW0ymy4OQWxmO_kuMiSf+3PkFonTujDy_PVMCvGYYg@mail.gmail.com | 
| Views: | Whole Thread | Raw Message | Download mbox | Resend email | 
| Thread: | |
| Lists: | pgsql-performance | 
On Tue, Oct 25, 2011 at 11:47 AM, Linos <info(at)linos(dot)es> wrote:
> El 25/10/11 18:43, Tom Lane escribió:
>> Linos <info(at)linos(dot)es> writes:
>>>     i am having any problems with performance of queries that uses CTE, can the
>>> join on a CTE use the index of the original table?
>>
>> CTEs act as optimization fences.  This is a feature, not a bug.  Use
>> them when you want to isolate the evaluation of a subquery.
>>
>>                       regards, tom lane
>>
>
> The truth it is that complex queries seems more readable using them (maybe a
> personal preference no doubt).
>
> Do have other popular databases the same behavior? SQL Server or Oracle for example?
In my experience, SQL server also materializes them -- basically CTE
is short hand for 'CREATE TEMP TABLE foo AS SELECT...' then joining to
foo.  If you want join behavior, use a join (by the way IIRC SQL
Server is a lot more restrictive about placement of ORDER BY).
I like CTE current behavior -- the main place I find it awkward is in
use of recursive queries because the CTE fence forces me to abstract
the recursion behind a function, not a view since pushing the view
qual down into the CTE is pretty horrible:
postgres=# explain select foo.id, (with bar as (select id from foo f
where f.id = foo.id) select * from bar) from foo where foo.id = 11;
                                     QUERY PLAN
-------------------------------------------------------------------------------------
 Index Scan using foo_idx on foo  (cost=0.00..16.57 rows=1 width=4)
   Index Cond: (id = 11)
   SubPlan 2
     ->  CTE Scan on bar  (cost=8.28..8.30 rows=1 width=4)
           CTE bar
             ->  Index Scan using foo_idx on foo f  (cost=0.00..8.28
rows=1 width=4)
                   Index Cond: (id = $0)
(7 rows)
whereas for function you can inject your qual inside the CTE pretty
easily.  this is a different problem than the one you're describing
though.  for the most part, CTE execution fence is a very good thing,
since it enforces restrictions that other features can leverage, for
example 'data modifying with' queries (by far my all time favorite
postgres enhancement).
merlin
| From | Date | Subject | |
|---|---|---|---|
| Next Message | Julius Tuskenis | 2011-10-26 07:48:33 | Re: how to use explain analyze | 
| Previous Message | Linos | 2011-10-25 16:47:53 | Re: CTE vs Subquery |