org.chocosolver.solver.variables.view.set.SetIntersectionView Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of choco-solver Show documentation
Show all versions of choco-solver Show documentation
Open-source constraint solver.
/*
* This file is part of choco-solver, http://choco-solver.org/
*
* Copyright (c) 2022, IMT Atlantique. All rights reserved.
*
* Licensed under the BSD 4-clause license.
*
* See LICENSE file in the project root for full license information.
*/
package org.chocosolver.solver.variables.view.set;
import org.chocosolver.solver.ICause;
import org.chocosolver.solver.exception.ContradictionException;
import org.chocosolver.solver.variables.SetVar;
import org.chocosolver.solver.variables.delta.ISetDelta;
import org.chocosolver.solver.variables.delta.ISetDeltaMonitor;
import org.chocosolver.solver.variables.events.IEventType;
import org.chocosolver.solver.variables.events.SetEventType;
import org.chocosolver.solver.variables.view.SetView;
import org.chocosolver.solver.variables.view.delta.SetViewOnSetsDeltaMonitor;
import org.chocosolver.util.objects.setDataStructures.ISet;
import org.chocosolver.util.objects.setDataStructures.SetFactory;
import org.chocosolver.util.objects.setDataStructures.SetType;
import org.chocosolver.util.objects.setDataStructures.dynamic.SetDifference;
import org.chocosolver.util.objects.setDataStructures.dynamic.SetIntersection;
import org.chocosolver.util.objects.setDataStructures.dynamic.SetUnion;
import org.chocosolver.util.procedure.IntProcedure;
/**
* Set view over set variables representing the intersection of these variables.
*
* @author Dimitri Justeau-Allaire
* @since 29/03/2021
*/
public class SetIntersectionView extends SetView {
protected ISet lb;
protected ISet ub;
protected ISet remove;
/**
* Create a set intersection view.
*
* @param name name of the variable
* @param variables observed variables
*/
public SetIntersectionView(String name, SetVar... variables) {
super(name, variables);
this.remove = SetFactory.makeStoredSet(SetType.RANGESET, 0, getModel());
ISet[] LBs = new ISet[variables.length];
ISet[] UBs = new ISet[variables.length];
for (int i = 0; i < variables.length; i++) {
LBs[i] = variables[i].getLB();
UBs[i] = variables[i].getUB();
}
this.lb = new SetIntersection(getModel(), LBs);
this.ub = new SetDifference(getModel(), new SetIntersection(getModel(), UBs), this.remove);
}
@Override
public ISet getLB() {
return lb;
}
@Override
public ISet getUB() {
return ub;
}
@Override
public boolean instantiateTo(int[] value, ICause cause) throws ContradictionException {
return false;
}
@Override
public boolean isInstantiated() {
return getLB().size() == getUB().size();
}
@Override
public void notify(IEventType event, int variableIdx) throws ContradictionException {
// Check if a multiple support removed element can effectively be removed
if ((event.getMask() & SetEventType.REMOVE_FROM_ENVELOPE.getMask()) > 0) {
for (int i : remove) {
if (doRemoveSetElement(i)) {
remove.remove(i);
break;
}
}
}
notifyPropagators(event, this);
}
@Override
protected boolean doRemoveSetElement(int element) throws ContradictionException {
// If there is only one observed set var containing the element, remove the element from this variable.
int nb = 0;
int idx = -1;
for (int i = 0; i < variables.length; i++) {
if (variables[i].getUB().contains(element)) {
nb++;
idx = i;
if (nb > 1) {
break;
}
}
}
if (nb == 1) {
return variables[idx].remove(element, this);
} else {
remove.add(element);
}
return false;
}
@Override
protected boolean doForceSetElement(int element) throws ContradictionException {
// Force the element in every set
boolean b = false;
for (SetVar set : variables) {
b |= set.force(element, this);
}
return b;
}
@Override
public ISetDelta getDelta() {
throw new UnsupportedOperationException("SetIntersectionView does not support getDelta()");
}
@Override
public ISetDeltaMonitor monitorDelta(ICause propagator) {
ISetDeltaMonitor[] deltaMonitors = new ISetDeltaMonitor[variables.length];
for (int i = 0; i < variables.length; i++) {
deltaMonitors[i] = variables[i].monitorDelta(propagator);
}
return new SetViewOnSetsDeltaMonitor(deltaMonitors) {
final ISet removed = SetFactory.makeStoredSet(SetType.RANGESET, 0, getModel());
final ISet remove = new SetDifference(new SetUnion(removedValues), removed);
final ISet add = new SetUnion(addedValues);
@Override
public void forEach(IntProcedure proc, SetEventType evt) throws ContradictionException {
fillValues();
if (evt == SetEventType.ADD_TO_KER) {
for (int v : add) {
if (getLB().contains(v)) {
proc.execute(v);
}
}
} else if (evt == SetEventType.REMOVE_FROM_ENVELOPE) {
for (int v : remove) {
proc.execute(v);
}
for (int v : remove) {
removed.add(v);
}
}
}
};
}
}