From 7cd5f147ea63fbead8d53927d52e144ff0e2e1a5 Mon Sep 17 00:00:00 2001 From: "Brian S. O'Neill" Date: Wed, 6 Sep 2006 01:34:09 +0000 Subject: Resolve orderings with unspecified direction. --- .../amazon/carbonado/qe/UnionQueryAnalyzer.java | 75 +++++++++++++++++----- 1 file changed, 60 insertions(+), 15 deletions(-) (limited to 'src/main/java/com/amazon/carbonado/qe') diff --git a/src/main/java/com/amazon/carbonado/qe/UnionQueryAnalyzer.java b/src/main/java/com/amazon/carbonado/qe/UnionQueryAnalyzer.java index 2906c8a..2f467eb 100644 --- a/src/main/java/com/amazon/carbonado/qe/UnionQueryAnalyzer.java +++ b/src/main/java/com/amazon/carbonado/qe/UnionQueryAnalyzer.java @@ -84,9 +84,34 @@ public class UnionQueryAnalyzer { return new Result(subResults); } - // FIXME: If any orderings have an unspecified direction, switch to - // ASCENDING or DESCENDING, depending on which is more popular. Then - // build new sub-results. + boolean canMutateOrderings = false; + + // If any orderings have an unspecified direction, switch to ASCENDING + // or DESCENDING, depending on which is more popular. Then build new + // sub-results. + for (int pos = 0; pos < orderings.size(); pos++) { + OrderedProperty ordering = orderings.get(pos); + if (ordering.getDirection() != Direction.UNSPECIFIED) { + continue; + } + + // Find out which direction is most popular for this property. + Tally tally = new Tally(ordering.getChainedProperty()); + for (IndexedQueryAnalyzer.Result result : subResults) { + tally.increment(findHandledDirection(result, ordering)); + } + + if (!canMutateOrderings) { + orderings = new ArrayList>(orderings); + canMutateOrderings = true; + } + + orderings.set(pos, ordering.direction(tally.getBestDirection())); + + // Re-calc with specified direction. Only do one property at a time + // since one simple change might alter the query plan. + subResults = splitIntoSubResults(filter, orderings); + } // Gather all the keys available. As ordering properties touch key // properties, they are removed from all key sets. When a key set size @@ -119,7 +144,10 @@ public class UnionQueryAnalyzer { } // Prepare to augment orderings to ensure a total ordering. - orderings = new ArrayList>(orderings); + if (!canMutateOrderings) { + orderings = new ArrayList>(orderings); + canMutateOrderings = true; + } // Keep looping until total ordering achieved. while (true) { @@ -143,14 +171,7 @@ public class UnionQueryAnalyzer { } } - // Find the best tally. - Tally best = null; - for (Tally tally : superKey.values()) { - if (best == null || tally.compareTo(best) < 0) { - best = tally; - } - } - + Tally best = bestTally(superKey.values()); ChainedProperty bestProperty = best.getProperty(); // Now augment the orderings and create new sub-results. @@ -217,6 +238,30 @@ public class UnionQueryAnalyzer { return result; } + private Tally bestTally(Iterable tallies) { + Tally best = null; + for (Tally tally : tallies) { + if (best == null || tally.compareTo(best) < 0) { + best = tally; + } + } + return best; + } + + private Direction findHandledDirection(IndexedQueryAnalyzer.Result result, + OrderedProperty unspecified) + { + ChainedProperty chained = unspecified.getChainedProperty(); + OrderingScore score = result.getCompositeScore().getOrderingScore(); + List> handled = score.getHandledOrderings(); + for (OrderedProperty property : handled) { + if (chained.equals(property)) { + return property.getDirection(); + } + } + return Direction.UNSPECIFIED; + } + private List.Result> splitIntoSubResults(Filter filter, List> orderings) { @@ -316,7 +361,7 @@ public class UnionQueryAnalyzer { } /** - * Used to track which properties should be augmented to create a total ordering. + * Used to track which property direction is most popular. */ private class Tally implements Comparable { private final ChainedProperty mProperty; @@ -353,14 +398,14 @@ public class UnionQueryAnalyzer { * Only returns ASCENDING or DESCENDING. */ Direction getBestDirection() { - if (mAscendingCount > mDescendingCount) { + if (mAscendingCount >= mDescendingCount) { return Direction.ASCENDING; } return Direction.DESCENDING; } int getBestCount() { - if (mAscendingCount > mDescendingCount) { + if (mAscendingCount >= mDescendingCount) { return mAscendingCount; } return mDescendingCount; -- cgit v1.2.3