All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.googlecode.cqengine.resultset.order.AttributeOrdersComparator Maven / Gradle / Ivy

Go to download

Collection Query Engine: NoSQL indexing and query engine for Java collections with ultra-low latency

There is a newer version: 3.6.0
Show newest version
/**
 * Copyright 2012-2015 Niall Gallagher
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.googlecode.cqengine.resultset.order;

import com.googlecode.cqengine.attribute.*;
import com.googlecode.cqengine.query.option.AttributeOrder;
import com.googlecode.cqengine.query.option.QueryOptions;

import java.util.Comparator;
import java.util.Iterator;
import java.util.List;

/**
 * A comparator which sorts result objects according to a list of attributes each with an associated preference for
 * ascending or descending order.
 *
 * @author Roberto Socrates
 * @author Niall Gallagher
 */
public class AttributeOrdersComparator implements Comparator {

    final List> attributeSortOrders;
    final QueryOptions queryOptions;

    public AttributeOrdersComparator(List> attributeSortOrders, QueryOptions queryOptions) {
        this.attributeSortOrders = attributeSortOrders;
        this.queryOptions = queryOptions;
    }

    @Override
    @SuppressWarnings("unchecked")
    public int compare(O o1, O o2) {
        for (AttributeOrder attributeOrder : attributeSortOrders) {
            Attribute attribute = attributeOrder.getAttribute();
            int comparison;
            if (attribute instanceof OrderControlAttribute) {
                OrderControlAttribute orderControl = (OrderControlAttribute)(OrderControlAttribute)(attribute);
                comparison = orderControl.getValue(o1, queryOptions).compareTo(orderControl.getValue(o2, queryOptions));
                if (comparison != 0) {
                    // One of the objects has values for the delegate attribute encapsulated in OrderControlAttribute,
                    // and the other object does not. Return this difference so that they will be ordered relative to
                    // each other based whether they have values or not...
                    return comparison;
                }
                attribute = (Attribute) orderControl.getDelegateAttribute();
            }
            comparison = compareAttributeValues(attribute, o1, o2);
            if (comparison != 0) {
                // Found a difference. Invert the sign if order is descending, and return it...
                return attributeOrder.isDescending() ? comparison * -1 : comparison;
            }
            // else continue checking remaining attributes.
        }
        // No differences found according to ordering specified, but in case this comparator
        // will be used for object equality testing, return 0 only if objects really are equal...
        if (o1.equals(o2)) {
            return 0;
        }

        // At this point we have run out of attributeSortOrders: they all returned 0, but because the objects are not
        // equal, we cannot return 0.

        // Example: This might occur when sorting by [color, price] and two or more different items have the same color
        // and the same price. Because a third - tie-breaking - sort order was not supplied, the sort order of items
        // which have the same color and price, is unspecified.

        // However although the sort order is now unspecified, we should try to preserve the stability of the comparison
        // as much as possible: such that
        //                              if comparator.compare(o1, o2) == -1, then it should be the case that
        //                                 comparator.compare(o2, o1) == +1.
        // Thus we cannot simply return the same result in both of those cases. This can be important when objects are
        // added to collections such as {@link java.util.TreeSet} which depend on the stability of the comparison.

        // As we don't have any other readily available properties of the objects to use as tie-breakers, we will use
        // the hashcodes of the objects instead. As the hashcodes of unequal objects are unlikely to be the same *most
        // of the time*, using the hashcodes as a tie-breaker will result in a stable comparison *most of the time*.
        // In the rare event that the hashcodes of the unequal objects turn out to be the same however, we will return
        // 1 because it seems more important that the comparator is consistent with equals() than it is to be stable...
        return o1.hashCode() >= o2.hashCode() ? 1 : -1;
    }

    > int compareAttributeValues(Attribute attribute, O o1, O o2) {
        if (attribute instanceof SimpleAttribute) {
            // Fast code path...
            SimpleAttribute simpleAttribute = (SimpleAttribute)attribute;
            return simpleAttribute.getValue(o1, queryOptions).compareTo(simpleAttribute.getValue(o2, queryOptions));
        }
        else {
            // Slower code path...
            Iterator o1attributeValues = attribute.getValues(o1, queryOptions).iterator();
            Iterator o2attributeValues = attribute.getValues(o2, queryOptions).iterator();
            while (o1attributeValues.hasNext() && o2attributeValues.hasNext()) {
                A o1attributeValue = o1attributeValues.next();
                A o2attributeValue = o2attributeValues.next();
                int comparison = o1attributeValue.compareTo(o2attributeValue);
                if (comparison != 0) {
                    // If we found a difference, return it...
                    return comparison;
                }
            }
            // If the number of attribute values differs, return a difference, object with fewest elements first...
            if (o2attributeValues.hasNext()) {
                return -1;
            }
            else if (o1attributeValues.hasNext()) {
                return +1;
            }
            // No differences found...
            return 0;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy