Thank you for the review,

On 27.11.2020 21:49, Heikki Linnakangas wrote:
On 31/10/2020 11:26, Andrey V. Lepikhov wrote:
+            /*
+             * Process restrictlist to seperate out the self join quals from +             * the other quals. e.g x = x goes to selfjoinquals and a = b to
+             * otherjoinquals.
+             */
+            split_selfjoin_quals(root, restrictlist, &selfjoinquals,
+                                 &otherjoinquals);
+
+            if (list_length(selfjoinquals) == 0)
+            {
+                /*
+                 * Have a chance to remove join if target list contains vars from
+                 * the only one relation.
+                 */

I don't understand the logic here. If 'selfjoinquals' is empty, it means that there is no join qual between the two relations, right? How can we ever remove the join in that case? And how does the target list affect that? Can you give an example query of that?

Maybe it is a problem of variable naming. Following the idea of David Rowley, we split quals into two subsets: {x==x} and another, for example {x=y}. First set is an trivial case of self-join: if we have unique index on the attribute 'x', then this join is self-join. Second set is give us a chance: if right side is unique for right side of the qual and no vars from right side end up in the target list of the join, then this is a self-join case. Example:

CREATE TABLE a(x int, y int);
CREATE UNIQUE INDEX ON a(x);
SELECT a1.* FROM a a1, a a2 WHERE a1.x = a2.x;  -- self-join
CREATE UNIQUE INDEX ON a(y);
SELECT a1.* FROM a a1, a a2 WHERE a1.x = a2.y;  -- self-join too


--- a/src/test/regress/expected/join.out
+++ b/src/test/regress/expected/join.out
@@ -4553,11 +4553,13 @@ explain (costs off)
 select p.* from
   (parent p left join child c on (p.k = c.k)) join parent x on p.k = x.k
   where p.k = 1 and p.k = 2;
-        QUERY PLAN ---------------------------
+                   QUERY PLAN +------------------------------------------------
  Result
    One-Time Filter: false
-(2 rows)
+   ->  Index Scan using parent_pkey on parent x
+         Index Cond: (k = 1)
+(4 rows)

 -- bug 5255: this is not optimizable by join removal
 begin;

That doesn't seem like an improvement...

I investigated this case. It is a planner feature: it simplifies dummy joins and dummy scans to different plans. Maybe it can cause some discussion, but this example so rare and doesn't make a problem.

My general impression of this patch is that it's a lot of code to deal with a rare special case. At the beginning of this thread there was discussion on the overhead that this might add to planning queries that don't benefit, but adding a lot of code isn't nice either, even if the performance is acceptable. That's not necessarily a show-stopper, but it does make me much less excited about this. I'm not sure what to suggest to do about that, except a really vague "Can you make is simpler?"

Currently planner reduces useless outer joins and unique semijoins. Reduce self join feature continues the development of the planner in the same direction. For example, it is needed for ORM software. Most of the code dedicated to removing unnecessary relation and replacing of one oid with another. We are trying to use remove_rel_from_query() machinery. Perhaps this will allow us to make the code shorter.

--
regards,
Andrey Lepikhov
Postgres Professional


Reply via email to