/*
* Sets.java - This file is part of the Jakstab project.
*
* Cross product method based on code taken from the MultiJava project.
* Copyright 1998-2002, Iowa State University
* Copyright 2007-2015 Johannes Kinder <jk@jakstab.org>
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, see <http://www.gnu.org/licenses/>.
*/
package org.jakstab.util;
import java.util.*;
import org.jakstab.util.Logger;
/**
* Helper functions for sets.
*
* @author Johannes Kinder
*/
public class Sets {
@SuppressWarnings("unused")
private static final Logger logger = Logger.getLogger(Sets.class);
public static final <T> Set<Tuple<T>> crossProduct(Tuple<Set<T>> tupleOfSets) {
return crossProductToIndex(tupleOfSets, tupleOfSets.size() - 1);
}
/*
* This functions deals automatically with RTLNumber.WILDCARD and ALL_NUMBERS, since it
* will assign the single element of ALL_NUMBERS, the WILDCARD, to all generated tuples
*/
private static final <T> Set<Tuple<T>> crossProductToIndex(Tuple<Set<T>> tupleOfSets, int pos) {
Set<Tuple<T>> setOfTuples = new FastSet<Tuple<T>>();
if (pos < 0) {
// we're generating the cross-product of 1 sets => return
// the singleton set of the empty tuple
setOfTuples.add(new Tuple<T>(0));
} else {
// first recursively generate the cross-product of all earlier
// positions
Set<Tuple<T>> setOfEarlierTuples =
crossProductToIndex(tupleOfSets, pos-1);
// then extend each of these for each element at this position
for (Tuple<T> earlierTuple : setOfEarlierTuples) {
for (T component : tupleOfSets.get(pos)) {
Tuple<T> newTuple = new Tuple<T>(pos+1);
for (int i = 0; i < pos; i++) {
newTuple.set(i, earlierTuple.get(i));
}
newTuple.set(pos, component);
setOfTuples.add(newTuple);
}
}
}
return setOfTuples;
}
}