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 Proposition
s.
* @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 extends TemporalProposition> 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 extends TemporalProposition> 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 extends TemporalProposition> 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 extends TemporalProposition> 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