-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpower_set.cpp
104 lines (94 loc) · 2.7 KB
/
power_set.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
#include <vector>
#include <span>
#include <iostream>
// Similar solution is in subsets_gen.cpp
template<typename T, typename U = typename std::remove_cv<T>::type>
static void generatePowerSets1(const std::span<T> array, std::size_t index, std::vector<U>& buffer, std::vector<std::vector<U>>& subsets)
{
if(index >= array.size())
{
subsets.push_back(buffer);
return;
}
buffer.push_back(array[index]);
generatePowerSets1(array, index + 1, buffer, subsets);
buffer.pop_back();
generatePowerSets1(array, index + 1, buffer, subsets);
}
template<typename T, typename U = typename std::remove_cv<T>::type>
static std::vector<std::vector<U>> generatePowerSets1(const std::span<T> array)
{
std::vector<U> buffer;
std::vector<std::vector<U>> subsets;
generatePowerSets1(array, 0, buffer, subsets);
return subsets;
}
template<typename T, typename U = typename std::remove_cv<T>::type>
static void addSubset(const std::span<T> array, std::size_t filter, std::vector<std::vector<U>>& subsets)
{
std::vector<U> set;
for(std::size_t i = 0; i < array.size(); ++i)
if(filter & (1ull << i))
set.push_back(array[i]);
subsets.push_back(std::move(set));
}
template<typename T, typename U = typename std::remove_cv<T>::type>
static std::vector<std::vector<U>> generatePowerSets2(const std::span<T> array)
{
std::vector<std::vector<U>> subsets;
std::size_t subsetCount = 1ull << array.size();
for(std::size_t i = 0; i < subsetCount; ++i)
addSubset(array, i, subsets);
return subsets;
}
template<typename T>
struct Solution1
{
auto operator()(const std::span<T> array)
{
return generatePowerSets1<T>(array);
}
};
template<typename T>
struct Solution2
{
auto operator()(const std::span<T> array)
{
return generatePowerSets2<T>(array);
}
};
template<typename T>
static std::ostream& operator<<(std::ostream& stream, const std::span<T> array)
{
stream << "{ ";
for(decltype(array.size()) i = 0; const auto& value : array)
{
stream << value;
if(++i < array.size())
stream << ", ";
}
stream << " }";
return stream;
}
template<template<typename> typename Sol, typename T>
static void runGeneratePowerSets(const std::span<T> array)
{
std::cout << "Input: " << array << "\n";
auto powerset = Sol<T> { } (array);
std::cout << "PowerSet cardinality: " << powerset.size() << "\n";
for(const auto& set : powerset)
std::cout << std::span{ set } << "\n";
}
template<typename T>
static void run(std::initializer_list<T> array)
{
std::cout << "**Solution no 1**\n";
runGeneratePowerSets<Solution1>(std::span { std::data(array), array.size() });
std::cout << "**Solution no 2**\n";
runGeneratePowerSets<Solution2>(std::span { std::data(array), array.size() });
}
int main()
{
run<int>({ 3, 8, 4, 1, 0, 10, 34, 56 });
return 0;
}