/* -*- C++ -*- * * This file is a part of LEMON, a generic C++ optimization library * * Copyright (C) 2003-2006 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport * (Egervary Research Group on Combinatorial Optimization, EGRES). * * Permission to use, modify and distribute this software is granted * provided that this copyright notice appears in all copies. For * precise terms see the accompanying LICENSE file. * * This software is provided "AS IS" with no warranty of any kind, * express or implied, and with no claim as to its suitability for any * purpose. * */ #ifndef LEMON_BITS_DEBUG_MAP_H #define LEMON_BITS_DEBUG_MAP_H #include #include #include #include #include #include #include #include ///\ingroup graphbits /// ///\file ///\brief Vector based graph maps for debugging. namespace lemon { /// \ingroup graphbits /// /// \brief Graph map based on the std::vector storage. /// /// The DebugMap template class is graph map structure what /// automatically updates the map when a key is added to or erased from /// the map. This map also checks some programming failures by example /// multiple addition of items, erasing of not existing item or /// not erased items at the destruction of the map. It helps the /// programmer to avoid segmentation faults and memory leaks. /// /// \param Notifier The AlterationNotifier that will notify this map. /// \param Item The item type of the graph items. /// \param Value The value type of the map. /// /// \author Balazs Dezso template class DebugMap : public ItemSetTraits<_Graph, _Item>::ItemNotifier::ObserverBase { private: /// The container type of the map. typedef std::vector<_Value> Container; /// The container type of the debug flags. typedef std::vector Flag; public: static const bool strictCheck = false; struct MapError { public: virtual ~MapError() {} virtual const char* what() const throw() { return "lemon::DebugMap::MapError"; } }; /// The graph type of the map. typedef _Graph Graph; /// The item type of the map. typedef _Item Item; /// The reference map tag. typedef True ReferenceMapTag; /// The key type of the map. typedef _Item Key; /// The value type of the map. typedef _Value Value; /// The notifier type. typedef typename ItemSetTraits<_Graph, _Item>::ItemNotifier Notifier; /// The map type. typedef DebugMap Map; /// The base class of the map. typedef typename Notifier::ObserverBase Parent; /// The reference type of the map; typedef typename Container::reference Reference; /// The const reference type of the map; typedef typename Container::const_reference ConstReference; /// \brief Constructor to attach the new map into the notifier. /// /// It constructs a map and attachs it into the notifier. /// It adds all the items of the graph to the map. DebugMap(const Graph& graph) { Parent::attach(graph.getNotifier(Item())); container.resize(Parent::getNotifier()->maxId() + 1); flag.resize(Parent::getNotifier()->maxId() + 1, false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { flag[Parent::getNotifier()->id(it)] = true; } } /// \brief Constructor uses given value to initialize the map. /// /// It constructs a map uses a given value to initialize the map. /// It adds all the items of the graph to the map. DebugMap(const Graph& graph, const Value& value) { Parent::attach(graph.getNotifier(Item())); container.resize(Parent::getNotifier()->maxId() + 1, value); flag.resize(Parent::getNotifier()->maxId() + 1, false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { flag[Parent::getNotifier()->id(it)] = true; } } /// \brief Copy constructor /// /// Copy constructor. DebugMap(const DebugMap& _copy) : Parent() { if (_copy.attached()) { Parent::attach(*_copy.getNotifier()); container = _copy.container; } flag.resize(Parent::getNotifier()->maxId() + 1, false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { flag[Parent::getNotifier()->id(it)] = true; LEMON_ASSERT(_copy.flag[Parent::getNotifier()->id(it)], MapError()); } } /// \brief Destructor /// /// Destructor. ~DebugMap() { const typename Parent::Notifier* notifier = Parent::getNotifier(); if (notifier != 0) { Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { LEMON_ASSERT(flag[Parent::getNotifier()->id(it)], MapError()); flag[Parent::getNotifier()->id(it)] = false; } } for (int i = 0; i < (int)flag.size(); ++i) { LEMON_ASSERT(!flag[i], MapError()); } } /// \brief Assign operator. /// /// This operator assigns for each item in the map the /// value mapped to the same item in the copied map. /// The parameter map should be indiced with the same /// itemset because this assign operator does not change /// the container of the map. DebugMap& operator=(const DebugMap& cmap) { return operator=(cmap); } /// \brief Template assign operator. /// /// The given parameter should be conform to the ReadMap /// concecpt and could be indiced by the current item set of /// the NodeMap. In this case the value for each item /// is assigned by the value of the given ReadMap. template DebugMap& operator=(const CMap& cmap) { checkConcept, CMap>(); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { set(it, cmap[it]); } return *this; } public: /// \brief The subcript operator. /// /// The subscript operator. The map can be subscripted by the /// actual items of the graph. Reference operator[](const Key& key) { LEMON_ASSERT(flag[Parent::getNotifier()->id(key)], MapError()); return container[Parent::getNotifier()->id(key)]; } /// \brief The const subcript operator. /// /// The const subscript operator. The map can be subscripted by the /// actual items of the graph. ConstReference operator[](const Key& key) const { LEMON_ASSERT(flag[Parent::getNotifier()->id(key)], MapError()); return container[Parent::getNotifier()->id(key)]; } /// \brief The setter function of the map. /// /// It the same as operator[](key) = value expression. void set(const Key& key, const Value& value) { (*this)[key] = value; } protected: /// \brief Adds a new key to the map. /// /// It adds a new key to the map. It called by the observer notifier /// and it overrides the add() member function of the observer base. virtual void add(const Key& key) { int id = Parent::getNotifier()->id(key); if (id >= (int)container.size()) { container.resize(id + 1); flag.resize(id + 1, false); } LEMON_ASSERT(!flag[Parent::getNotifier()->id(key)], MapError()); flag[Parent::getNotifier()->id(key)] = true; if (strictCheck) { std::vector fl(flag.size(), false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { int id = Parent::getNotifier()->id(it); fl[id] = true; } LEMON_ASSERT(fl == flag, MapError()); } } /// \brief Adds more new keys to the map. /// /// It adds more new keys to the map. It called by the observer notifier /// and it overrides the add() member function of the observer base. virtual void add(const std::vector& keys) { int max = container.size() - 1; for (int i = 0; i < (int)keys.size(); ++i) { int id = Parent::getNotifier()->id(keys[i]); if (id >= max) { max = id; } } container.resize(max + 1); flag.resize(max + 1, false); for (int i = 0; i < (int)keys.size(); ++i) { LEMON_ASSERT(!flag[Parent::getNotifier()->id(keys[i])], MapError()); flag[Parent::getNotifier()->id(keys[i])] = true; } if (strictCheck) { std::vector fl(flag.size(), false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { int id = Parent::getNotifier()->id(it); fl[id] = true; } LEMON_ASSERT(fl == flag, MapError()); } } /// \brief Erase a key from the map. /// /// Erase a key from the map. It called by the observer notifier /// and it overrides the erase() member function of the observer base. virtual void erase(const Key& key) { if (strictCheck) { std::vector fl(flag.size(), false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { int id = Parent::getNotifier()->id(it); fl[id] = true; } LEMON_ASSERT(fl == flag, MapError()); } container[Parent::getNotifier()->id(key)] = Value(); LEMON_ASSERT(flag[Parent::getNotifier()->id(key)], MapError()); flag[Parent::getNotifier()->id(key)] = false; } /// \brief Erase more keys from the map. /// /// Erase more keys from the map. It called by the observer notifier /// and it overrides the erase() member function of the observer base. virtual void erase(const std::vector& keys) { if (strictCheck) { std::vector fl(flag.size(), false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { int id = Parent::getNotifier()->id(it); fl[id] = true; } LEMON_ASSERT(fl == flag, MapError()); } for (int i = 0; i < (int)keys.size(); ++i) { container[Parent::getNotifier()->id(keys[i])] = Value(); LEMON_ASSERT(flag[Parent::getNotifier()->id(keys[i])], MapError()); flag[Parent::getNotifier()->id(keys[i])] = false; } } /// \brief Buildes the map. /// /// It buildes the map. It called by the observer notifier /// and it overrides the build() member function of the observer base. virtual void build() { if (strictCheck) { for (int i = 0; i < (int)flag.size(); ++i) { LEMON_ASSERT(flag[i], MapError()); } } int size = Parent::getNotifier()->maxId() + 1; container.reserve(size); container.resize(size); flag.reserve(size); flag.resize(size, false); const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { int id = Parent::getNotifier()->id(it); LEMON_ASSERT(!flag[id], MapError()); flag[id] = true; } } /// \brief Clear the map. /// /// It erase all items from the map. It called by the observer notifier /// and it overrides the clear() member function of the observer base. virtual void clear() { const typename Parent::Notifier* notifier = Parent::getNotifier(); Item it; for (notifier->first(it); it != INVALID; notifier->next(it)) { int id = Parent::getNotifier()->id(it); LEMON_ASSERT(flag[id], MapError()); flag[id] = false; } if (strictCheck) { for (int i = 0; i < (int)flag.size(); ++i) { LEMON_ASSERT(!flag[i], MapError()); } } container.clear(); flag.clear(); } private: Container container; Flag flag; }; } #endif