[2df580] | 1 | /*
|
---|
| 2 | * SubsetMap.hpp
|
---|
| 3 | *
|
---|
| 4 | * Created on: Jul 3, 2012
|
---|
| 5 | * Author: heber
|
---|
| 6 | */
|
---|
| 7 |
|
---|
| 8 | #ifndef SUBSETMAP_HPP_
|
---|
| 9 | #define SUBSETMAP_HPP_
|
---|
| 10 |
|
---|
| 11 |
|
---|
| 12 | // include config.h
|
---|
| 13 | #ifdef HAVE_CONFIG_H
|
---|
| 14 | #include <config.h>
|
---|
| 15 | #endif
|
---|
| 16 |
|
---|
| 17 | #include <boost/shared_ptr.hpp>
|
---|
| 18 | #include <map>
|
---|
| 19 |
|
---|
| 20 | #include "CodePatterns/Assert.hpp"
|
---|
| 21 |
|
---|
| 22 | #include "IndexSetContainer.hpp"
|
---|
| 23 |
|
---|
| 24 | class SubsetMapTest;
|
---|
| 25 |
|
---|
| 26 | /** A SubsetMap is a map containing shared_ptr of IndexSet instances that knows
|
---|
| 27 | * about which set is contained in which for fast lookup for SubsetValues.
|
---|
| 28 | */
|
---|
| 29 | class SubsetMap
|
---|
| 30 | {
|
---|
| 31 | //!> grant unit test access
|
---|
| 32 | friend class SubsetMapTest;
|
---|
| 33 | public:
|
---|
| 34 | //!> typedef for wrapping an instance in a shared_ptr
|
---|
| 35 | typedef boost::shared_ptr<SubsetMap> ptr;
|
---|
| 36 |
|
---|
| 37 | /** Constructor for SubsetMap that creates the internal map from IndexSet to a
|
---|
| 38 | * container with all contained sets.
|
---|
| 39 | *
|
---|
| 40 | * To this end, we first put each IndexSet in a multimap from size to IndexSet.
|
---|
| 41 | * Then we go "level-wise" from low to high through this map and use gatherSubsets()
|
---|
| 42 | * on each set as we are then sure that all contained subsets have already been
|
---|
| 43 | * initialized.
|
---|
| 44 | *
|
---|
| 45 | * @param _container container with IndexSet
|
---|
| 46 | */
|
---|
| 47 | SubsetMap(IndexSetContainer &_container);
|
---|
| 48 |
|
---|
| 49 | /** Getter for all subsets of set \a ptr.
|
---|
| 50 | *
|
---|
| 51 | * @param ptr IndexSet
|
---|
| 52 | * @return IndexSetContainer with all contained sets
|
---|
| 53 | */
|
---|
| 54 | IndexSetContainer::ptr & getSubsets(const IndexSet_ptr &ptr) {
|
---|
| 55 | Lookup_t::iterator lookupiter = Lookup.find(ptr);
|
---|
| 56 | ASSERT(lookupiter != Lookup.end(),
|
---|
| 57 | "SubsetMap::getSubsets() - the set "+toString(*ptr)+" is not in the Lookup.");
|
---|
| 58 | return lookupiter->second;
|
---|
| 59 | }
|
---|
| 60 |
|
---|
[db11d4] | 61 | /** Returns the size of the largest \b sub set.
|
---|
| 62 | *
|
---|
| 63 | * This would be \f${\cal O}(n)\f$ if we had to look at each set. However, due
|
---|
| 64 | * to the specific sorting we just have to check the last sets.
|
---|
| 65 | *
|
---|
| 66 | * @return number of indices in largest subset, -1 if SubsetMap contains no IndexSet's
|
---|
| 67 | */
|
---|
| 68 | size_t getMaximumSubsetLevel() const;
|
---|
| 69 |
|
---|
[d199cc] | 70 | /** Returns the size of the largest set.
|
---|
| 71 | *
|
---|
| 72 | * @return number of indices in largest set
|
---|
| 73 | */
|
---|
| 74 | size_t getMaximumSetLevel() const;
|
---|
| 75 |
|
---|
[2df580] | 76 | private:
|
---|
| 77 | /** Private function to fill the internal Lookup for a given IndexSet \a ptr.
|
---|
| 78 | *
|
---|
| 79 | * @param ptr IndexSet
|
---|
| 80 | */
|
---|
| 81 | void gatherSubsets(const IndexSet_ptr &ptr);
|
---|
| 82 |
|
---|
| 83 | /** Returns the index of \a subset in the power subsets of \a ptr.
|
---|
| 84 | *
|
---|
| 85 | * @param ptr set to check
|
---|
| 86 | * @param subset subset whose lexicographical position to find in \a ptr
|
---|
| 87 | * @return index of this subset
|
---|
| 88 | */
|
---|
| 89 | static const size_t getPowerSetIndex(const IndexSet_ptr &ptr, const IndexSet &subset);
|
---|
| 90 |
|
---|
| 91 | /** Returns the number of power subsets for a given set size.
|
---|
| 92 | *
|
---|
| 93 | * @param SetSize size of the set
|
---|
| 94 | * @return possible number of true subsets
|
---|
| 95 | */
|
---|
| 96 | static size_t getNoPowerSets(const size_t SetSize) {
|
---|
| 97 | return (1 << SetSize);
|
---|
| 98 | }
|
---|
| 99 |
|
---|
| 100 | /** Returns the subset for a given \a _set by its power set index \a PowerSetNo.
|
---|
| 101 | *
|
---|
| 102 | * The idea is that each bit in \a PowerSetNo encodes whether the Index at the
|
---|
| 103 | * position in the set is to appear in the subset or not. Hence, we use std::bitset
|
---|
| 104 | * to construct the bit representation from \a PowerSetNo and then go through each
|
---|
| 105 | * bit and either insert into the subset or not.
|
---|
| 106 | *
|
---|
| 107 | * @param _set index set
|
---|
| 108 | * @param PowerSetNo subset index
|
---|
| 109 | * @return specific power set subset given by the index
|
---|
| 110 | */
|
---|
| 111 | static IndexSet getSubset(const IndexSet &_set, size_t PowerSetNo);
|
---|
| 112 |
|
---|
| 113 | private:
|
---|
| 114 | //!> enumeration of states whether a subset is contained or not
|
---|
| 115 | enum ContainedState {
|
---|
| 116 | NotContained = 0,
|
---|
| 117 | Contained = 1,
|
---|
| 118 | };
|
---|
| 119 |
|
---|
| 120 | //!> temporary IndexSet_ptr
|
---|
| 121 | IndexSet_ptr tempset;
|
---|
| 122 | //!> enumeration trick to define a maximum subset size
|
---|
| 123 | enum {MAX_SETSIZE = 32};
|
---|
| 124 | //!> typedef for the specific map from IndexSet to all contained IndexSets
|
---|
| 125 | typedef std::map< IndexSet_ptr, IndexSetContainer::ptr,
|
---|
| 126 | IndexSetContainer::Comparator_t> Lookup_t;
|
---|
| 127 | Lookup_t Lookup;
|
---|
| 128 | };
|
---|
| 129 |
|
---|
| 130 | #endif /* SUBSETMAP_HPP_ */
|
---|