org.optaplanner.examples.common.experimental.impl.IntervalClusterImpl Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of optaplanner-examples Show documentation
Show all versions of optaplanner-examples Show documentation
OptaPlanner solves planning problems.
This lightweight, embeddable planning engine implements powerful and scalable algorithms
to optimize business resource scheduling and planning.
This module contains the examples which demonstrate how to use it in a normal Java application.
package org.optaplanner.examples.common.experimental.impl;
import java.util.Iterator;
import java.util.NavigableSet;
import java.util.function.BiFunction;
import org.optaplanner.examples.common.experimental.api.IntervalCluster;
final class IntervalClusterImpl, Difference_ extends Comparable>
implements IntervalCluster {
private final NavigableSet> splitPointSet;
private final BiFunction differenceFunction;
private IntervalSplitPoint startSplitPoint;
private IntervalSplitPoint endSplitPoint;
private int count;
private boolean hasOverlap;
IntervalClusterImpl(NavigableSet> splitPointSet,
BiFunction differenceFunction, IntervalSplitPoint start) {
if (start == null) {
throw new IllegalArgumentException("start (" + start + ") is null");
}
if (differenceFunction == null) {
throw new IllegalArgumentException("differenceFunction (" + differenceFunction + ") is null");
}
this.splitPointSet = splitPointSet;
this.startSplitPoint = start;
this.endSplitPoint = start;
this.differenceFunction = differenceFunction;
int activeIntervals = 0;
count = 0;
boolean anyOverlap = false;
IntervalSplitPoint current = start;
do {
count += current.intervalsStartingAtSplitPointSet.size();
activeIntervals += current.intervalsStartingAtSplitPointSet.size() - current.intervalsEndingAtSplitPointSet.size();
if (activeIntervals > 1) {
anyOverlap = true;
}
current = splitPointSet.higher(current);
} while (activeIntervals > 0 && current != null);
hasOverlap = anyOverlap;
if (current != null) {
endSplitPoint = splitPointSet.lower(current);
} else {
endSplitPoint = splitPointSet.last();
}
}
IntervalClusterImpl(NavigableSet> splitPointSet,
BiFunction differenceFunction, IntervalSplitPoint start,
IntervalSplitPoint end, int count, boolean hasOverlap) {
this.splitPointSet = splitPointSet;
this.startSplitPoint = start;
this.endSplitPoint = end;
this.differenceFunction = differenceFunction;
this.count = count;
this.hasOverlap = hasOverlap;
}
IntervalSplitPoint getStartSplitPoint() {
return startSplitPoint;
}
IntervalSplitPoint getEndSplitPoint() {
return endSplitPoint;
}
void addInterval(Interval interval) {
if (interval.getEndSplitPoint().compareTo(getStartSplitPoint()) > 0
&& interval.getStartSplitPoint().compareTo(getEndSplitPoint()) < 0) {
hasOverlap = true;
}
if (interval.getStartSplitPoint().compareTo(startSplitPoint) < 0) {
startSplitPoint = splitPointSet.floor(interval.getStartSplitPoint());
}
if (interval.getEndSplitPoint().compareTo(endSplitPoint) > 0) {
endSplitPoint = splitPointSet.ceiling(interval.getEndSplitPoint());
}
count++;
}
Iterable> removeInterval(Interval interval) {
return IntervalClusterIterator::new;
}
void mergeIntervalCluster(IntervalClusterImpl laterIntervalCluster) {
if (endSplitPoint.compareTo(laterIntervalCluster.startSplitPoint) > 0) {
hasOverlap = true;
}
if (endSplitPoint.compareTo(laterIntervalCluster.endSplitPoint) < 0) {
endSplitPoint = laterIntervalCluster.endSplitPoint;
}
count += laterIntervalCluster.count;
hasOverlap |= laterIntervalCluster.hasOverlap;
}
@Override
public Iterator iterator() {
return new IntervalTreeIterator<>(splitPointSet.subSet(startSplitPoint, true, endSplitPoint, true));
}
@Override
public int size() {
return count;
}
@Override
public boolean hasOverlap() {
return hasOverlap;
}
@Override
public Point_ getStart() {
return startSplitPoint.splitPoint;
}
@Override
public Point_ getEnd() {
return endSplitPoint.splitPoint;
}
@Override
public Difference_ getLength() {
return differenceFunction.apply(startSplitPoint.splitPoint, endSplitPoint.splitPoint);
}
@Override
public String toString() {
return "IntervalCluster{" +
"startSplitPoint=" + startSplitPoint +
", endSplitPoint=" + endSplitPoint +
", count=" + count +
", hasOverlap=" + hasOverlap +
'}';
}
// TODO: Make this incremental by only checking between the interval's start and end points
private final class IntervalClusterIterator
implements Iterator> {
private IntervalSplitPoint current = getStart(startSplitPoint);
private IntervalSplitPoint
getStart(IntervalSplitPoint start) {
while (start != null && start.isEmpty()) {
start = splitPointSet.higher(start);
}
return start;
}
@Override
public boolean hasNext() {
return current != null && current.compareTo(endSplitPoint) <= 0 && !splitPointSet.isEmpty();
}
@Override
public IntervalClusterImpl next() {
IntervalSplitPoint start = current;
IntervalSplitPoint end;
int activeIntervals = 0;
count = 0;
boolean anyOverlap = false;
do {
count += current.intervalsStartingAtSplitPointSet.size();
activeIntervals +=
current.intervalsStartingAtSplitPointSet.size() - current.intervalsEndingAtSplitPointSet.size();
if (activeIntervals > 1) {
anyOverlap = true;
}
current = splitPointSet.higher(current);
} while (activeIntervals > 0 && current != null);
hasOverlap = anyOverlap;
if (current != null) {
end = splitPointSet.lower(current);
current = getStart(current);
} else {
end = splitPointSet.last();
}
return new IntervalClusterImpl<>(splitPointSet, differenceFunction, start, end, count, hasOverlap);
}
}
}