/* * * Copyright (C) 2015 Jason Gowan * All rights reserved. * * This software may be modified and distributed under the terms * of the BSD license. See the LICENSE file for details. */ #ifndef COMBINATIONS_H_ #define COMBINATIONS_H_ #include #include #include "dither_types.h" namespace dither { inline void product( std::forward_list& rvvi, // final result dtest_case& rvi, // current result std::vector::const_iterator me, // current input std::vector::const_iterator end); // final input inline void product2( std::forward_list>& rvvi, // final result std::vector& rvi, // current result std::vector>::const_iterator me, // current input std::vector>::const_iterator end); // final input inline void gcombinations_(const std::vector&, std::vector&, std::forward_list>&); inline void combinations(const int, const std::vector&, std::forward_list>&); inline void product2( std::forward_list>& rvvi, // final result std::vector& rvi, // current result std::vector>::const_iterator me, // current input std::vector>::const_iterator end) // final input { if (me == end) { // terminal condition of the recursion. We no longer have // any input vectors to manipulate. Add the current result (rvi) // to the total set of results (rvvvi). rvvi.push_front(rvi); return; } // need an easy name for my vector-of-ints const std::vector& mevi = *me; for (std::vector::const_iterator it = mevi.begin(); it != mevi.end(); it++) { // final rvi will look like "a, b, c, ME, d, e, f" // At the moment, rvi already has "a, b, c" rvi.push_back(*it); // add ME product2(rvvi, rvi, me + 1, end); // add "d, e, f" rvi.pop_back(); // clean ME off for next round } } void combinations(const int n, const std::vector& input, std::forward_list>& output) { std::vector scratch(n); if (n <= input.size()) { for (int i = 0; i < n; i++) { scratch[i] = i; } gcombinations_(input, scratch, output); while (1) { int i; for (i = n - 1; (i >= 0) && (scratch[i] == (input.size() - n + i)); i--) ; if (i < 0) { break; } else { scratch[i]++; for (++i; i < n; i++) { scratch[i] = scratch[i - 1] + 1; } gcombinations_(input, scratch, output); } } } } inline void gcombinations_(const std::vector& input, std::vector& scratch, std::forward_list>& output) { std::vector result(scratch.size()); for (std::size_t i = 0; i < result.size(); i++) { result[i] = input[scratch[i]]; } output.push_front(result); } inline void product( std::forward_list& rvvi, // final result dtest_case& rvi, // current result std::vector::const_iterator me, // current input std::vector::const_iterator end) // final input { if (me == end) { // terminal condition of the recursion. We no longer have // any input vectors to manipulate. Add the current result (rvi) // to the total set of results (rvvvi). rvvi.push_front(rvi); return; } // need an easy name for my vector-of-ints const dtest_case& mevi = *me; for (dtest_case::const_iterator it = mevi.begin(); it != mevi.end(); it++) { // final rvi will look like "a, b, c, ME, d, e, f" // At the moment, rvi already has "a, b, c" rvi.push_back(*it); // add ME product(rvvi, rvi, me + 1, end); // add "d, e, f" rvi.pop_back(); // clean ME off for next round } } } #endif