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

org.protempa.proposition.PropositionUtil Maven / Gradle / Ivy

/*
 * #%L
 * Protempa Framework
 * %%
 * Copyright (C) 2012 - 2013 Emory University
 * %%
 * 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.
 * #L%
 */
package org.protempa.proposition;

import java.util.ArrayList;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.RandomAccess;
import org.protempa.proposition.comparator.MaxFinishTemporalPropositionComparator;
import org.protempa.proposition.comparator.TemporalPropositionIntervalComparator;

import org.protempa.proposition.interval.IntervalUtil;
import org.protempa.proposition.value.Unit;

/**
 * @author Andrew Post
 */
public class PropositionUtil {

    private PropositionUtil() {
    }

    /**
     * Filters a list of temporal propositions by time.
     *
     * @param params
     *            a List of propositions.
     * @param minValid
     * @param maxValid
     * @return
     */
    public static  List getView(
            List params, Long minValid, Long maxValid) {
        if (params != null) {
            if (minValid == null && maxValid == null) {
                return params;
            } else {
                int min = minValid != null ? binarySearchMinStart(params,
                        minValid.longValue()) : 0;
                int max = maxValid != null ? binarySearchMaxFinish(params,
                        maxValid.longValue()) : params.size();
                return params.subList(min, max);
            }
        } else {
            return null;
        }
    }

    public static  Map> getView(
            Map> paramsByKey, Long minValid, Long maxValid) {
        if (paramsByKey != null) {
            Map> result = new HashMap<>();
            for (Map.Entry> me : paramsByKey.entrySet()) {
                result.put(me.getKey(), getView(me.getValue(), minValid,
                        maxValid));
            }
            return result;
        } else {
            return null;
        }
    }

    /**
     * Divides a list of propositions up by id.
     *
     * @param propositions
     *            a List of Propositions.
     * @return a Map of id String -> List, with
     *         propositions in the same order as they were found in the
     *         argument.
     */
    public static  Map> createPropositionMap(
            List propositions) {
        Map> result = new HashMap<>();

        if (propositions != null) {
            for (T prop : propositions) {
                String propId = prop.getId();

                List ts = null;
                if (result.containsKey(propId)) {
                    ts = result.get(propId);
                } else {
                    ts = new ArrayList<>();
                    result.put(propId, ts);
                }
                ts.add(prop);
            }
        }

        return result;
    }

    /**
     * Binary search for a primitive parameter by timestamp, optimized for when
     * the parameters are stored in a list that implements
     * java.util.RandomAccess.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if
     *         not found.
     */
    private static int minStartIndexedBinarySearch(
            List list, long tstamp) {
        int low = 0;
        int high = list.size() - 1;

        while (low <= high) {
            /*
             * We use >>> instead of >> or /2 to avoid overflow. Sun's
             * implementation of binary search actually doesn't do this (bug
             * #5045582).
             */
            int mid = (low + high) >>> 1;
            TemporalProposition midVal = list.get(mid);
            Long minStart = midVal.getInterval().getMinimumStart();
            int cmp = minStart != null ? minStart.compareTo(tstamp) : -1;
            if (cmp < 0) {
                low = mid + 1;
            } else if (cmp > 0) {
                high = mid - 1;
            } else {
                return mid;
            }
        }
        return low;
    }

    /**
     * Binary search for a primitive parameter by timestamp, optimized for when
     * the parameters are stored in a list that does not implement
     * java.util.RandomAccess.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if
     *         not found.
     */
    private static  int minStartIteratorBinarySearch(
            List list, long tstamp) {
        int low = 0;
        int high = list.size() - 1;
        ListIterator i = list.listIterator();

        while (low <= high) {
            /*
             * We use >>> instead of >> or /2 to avoid overflow. Sun's
             * implementation of binary search actually doesn't do this (bug
             * #5045582).
             */
            int mid = (low + high) >>> 1;
            TemporalProposition midVal = iteratorBinarySearchGet(i, mid);
            Long maxStart = midVal.getInterval().getMinimumStart();
            int cmp = maxStart != null ? maxStart.compareTo(tstamp) : 1;
            if (cmp < 0) {
                low = mid + 1;
            } else if (cmp > 0) {
                high = mid - 1;
            } else {
                return mid;
            }
        }
        return low;
    }

    /**
     * Binary search for a primitive parameter by timestamp.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if not found.
     */
    private static int binarySearchMinStart(
            List params, long timestamp) {
        /*
         * The conditions for using index versus iterator are grabbed from the
         * JDK source code.
         */
        if (params.size() < 5000 || params instanceof RandomAccess) {
            return minStartIndexedBinarySearch(params, timestamp);
        } else {
            return minStartIteratorBinarySearch(params, timestamp);
        }
    }

    /**
     * Binary search for a primitive parameter by timestamp, optimized for when
     * the parameters are stored in a list that implements
     * java.util.RandomAccess.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if
     *         not found.
     */
    private static int maxFinishIndexedBinarySearch(
            List list, long tstamp) {
        int low = 0;
        int high = list.size() - 1;

        while (low <= high) {
            /*
             * We use >>> instead of >> or /2 to avoid overflow. Sun's
             * implementation of binary search actually doesn't do this (bug
             * #5045582).
             */
            int mid = (low + high) >>> 1;
            TemporalProposition midVal = list.get(mid);
            Long maxFinish = midVal.getInterval().getMaximumFinish();
            int cmp = maxFinish != null ? maxFinish.compareTo(tstamp) : 1;
            if (cmp < 0) {
                low = mid + 1;
            } else if (cmp > 0) {
                high = mid - 1;
            } else {
                return mid + 1;
            }
        }

        return high + 1;
    }

    /**
     * Binary search for a primitive parameter by timestamp, optimized for when
     * the parameters are stored in a list that does not implement
     * java.util.RandomAccess.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if
     *         not found.
     */
    private static  int maxFinishIteratorBinarySearch(
            List list, long tstamp) {
        int low = 0;
        int high = list.size() - 1;
        ListIterator i = list.listIterator();

        while (low <= high) {
            /*
             * We use >>> instead of >> or /2 to avoid overflow. Sun's
             * implementation of binary search actually doesn't do this (bug
             * #5045582).
             */
            int mid = (low + high) >>> 1;
            T midVal = iteratorBinarySearchGet(i, mid);
            Long maxFinish = midVal.getInterval().getMaximumFinish();
            int cmp = maxFinish != null ? maxFinish.compareTo(tstamp) : 1;
            if (cmp < 0) {
                low = mid + 1;
            } else if (cmp > 0) {
                high = mid - 1;
            } else {
                return mid + 1;
            }
        }
        return high + 1;
    }

    /**
     * Binary search for a primitive parameter by timestamp.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if not found.
     */
    private static int binarySearchMaxFinish(
            List params, long timestamp) {
        /*
         * The conditions for using index versus iterator are grabbed from the
         * JDK source code.
         */
        if (params.size() < 5000 || params instanceof RandomAccess) {
            return maxFinishIndexedBinarySearch(params, timestamp);
        } else {
            return maxFinishIteratorBinarySearch(params, timestamp);
        }
    }

    /*
     * The following four methods implement binary search for a primitive
     * parameter with a given timestamp. We don't use the JDK's built-in binary
     * search because we'd have to generate a new list containing just the
     * parameters' timestamps, search that for the timestamp of interest,
     * retrieve the index and grab the primitive parameter with that index. Just
     * seemed more efficient and not much more complicated to write our own.
     */
    /**
     * Binary search for a primitive parameter by timestamp, optimized for when
     * the parameters are stored in a list that implements
     * java.util.RandomAccess.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if
     *         not found.
     */
    private static PrimitiveParameter indexedBinarySearch(
            List list, long tstamp) {
        int low = 0;
        int high = list.size() - 1;

        while (low <= high) {
            /*
             * We use >>> instead of >> or /2 to avoid overflow. Sun's
             * implementation of binary search actually doesn't do this (bug
             * #5045582).
             */
            int mid = (low + high) >>> 1;
            PrimitiveParameter midVal = list.get(mid);
            long cmp = midVal.getPosition() - tstamp;

            if (cmp < 0) {
                low = mid + 1;
            } else if (cmp > 0) {
                high = mid - 1;
            } else {
                return midVal;
            }
        }
        return null;
    }

    /**
     * Binary search for a primitive parameter by timestamp, optimized for when
     * the parameters are stored in a list that does not implement
     * java.util.RandomAccess.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if
     *         not found.
     */
    private static  PrimitiveParameter iteratorBinarySearch(
            List list, long tstamp) {
        int low = 0;
        int high = list.size() - 1;
        ListIterator i = list.listIterator();

        while (low <= high) {
            /*
             * We use >>> instead of >> or /2 to avoid overflow. Sun's
             * implementation of binary search actually doesn't do this (bug
             * #5045582).
             */
            int mid = (low + high) >>> 1;
            PrimitiveParameter midVal = iteratorBinarySearchGet(i, mid);
            long cmp = midVal.getPosition() - tstamp;

            if (cmp < 0) {
                low = mid + 1;
            } else if (cmp > 0) {
                high = mid - 1;
            } else {
                return midVal;
            }
        }
        return null;
    }

    /**
     * Gets the ith element from a list by repositioning the specified list's
     * ListIterator.
     *
     * @param i
     *            a ListIterator.
     * @param index
     *            an index.
     */
    private static  T iteratorBinarySearchGet(ListIterator i, int index) {
        T obj = null;
        int pos = i.nextIndex();
        if (pos <= index) {
            do {
                obj = i.next();
            } while (pos++ < index);
        } else {
            do {
                obj = i.previous();
            } while (--pos > index);
        }
        return obj;
    }

    /**
     * Binary search for a primitive parameter by timestamp.
     *
     * @param list
     *            a List of PrimitiveParameter
     *            objects all with the same paramId, cannot be null.
     * @param tstamp
     *            the timestamp we're interested in finding.
     * @return a PrimitiveParameter, or null if not found.
     */
    public static PrimitiveParameter binarySearch(
            List params, long timestamp) {
        /*
         * The conditions for using index versus iterator are grabbed from the
         * JDK source code.
         */
        if (params.size() < 5000 || params instanceof RandomAccess) {
            return indexedBinarySearch(params, timestamp);
        } else {
            return iteratorBinarySearch(params, timestamp);
        }
    }
    
    public static long distanceBetween(TemporalProposition firstProp,
            TemporalProposition secondProp) {
        return distanceBetween(firstProp, secondProp, null);
    }
    
    public static long distanceBetween(TemporalProposition firstProp,
            TemporalProposition secondProp, Unit units) {
        if (firstProp == null) {
            throw new IllegalArgumentException("firstProp cannot be null");
        }
        if (secondProp == null) {
            throw new IllegalArgumentException("secondProp cannot be null");
        }
        return IntervalUtil.distanceBetween(firstProp.getInterval(), 
                secondProp.getInterval(), units);
    }
    
    public static String distanceBetweenFormattedShort(
            TemporalProposition firstProp, TemporalProposition secondProp) {
        return distanceBetweenFormattedShort(firstProp, secondProp, null);
    }
    
    public static String distanceBetweenFormattedShort(
            TemporalProposition firstProp, TemporalProposition secondProp, 
            Unit units) {
        if (firstProp == null) {
            throw new IllegalArgumentException("firstProp cannot be null");
        }
        if (secondProp == null) {
            throw new IllegalArgumentException("secondProp cannot be null");
        }
        return IntervalUtil.distanceBetweenFormattedShort(firstProp.getInterval(), 
                secondProp.getInterval(), units);
    }
    
    public static String distanceBetweenFormattedMedium(
            TemporalProposition firstProp, TemporalProposition secondProp) {
        return distanceBetweenFormattedMedium(firstProp, secondProp, null);
    }
    
    public static String distanceBetweenFormattedMedium(
            TemporalProposition firstProp, TemporalProposition secondProp, 
            Unit units) {
        if (firstProp == null) {
            throw new IllegalArgumentException("firstProp cannot be null");
        }
        if (secondProp == null) {
            throw new IllegalArgumentException("secondProp cannot be null");
        }
        return IntervalUtil.distanceBetweenFormattedMedium(
                firstProp.getInterval(), secondProp.getInterval(), units);
    }
    
    public static String distanceBetweenFormattedLong(
            TemporalProposition firstProp, TemporalProposition secondProp) {
        return distanceBetweenFormattedLong(firstProp, secondProp, null);
    }
    
    public static String distanceBetweenFormattedLong(
            TemporalProposition firstProp, TemporalProposition secondProp, 
            Unit units) {
        if (firstProp == null) {
            throw new IllegalArgumentException("firstProp cannot be null");
        }
        if (secondProp == null) {
            throw new IllegalArgumentException("secondProp cannot be null");
        }
        return IntervalUtil.distanceBetweenFormattedLong(
                firstProp.getInterval(), secondProp.getInterval(), units);
    }

    static final Comparator TEMP_PROP_COMP =
            new TemporalPropositionIntervalComparator();

    static final Comparator MAX_FINISH_COMP =
            new MaxFinishTemporalPropositionComparator();
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy