7 /// \brief Bfs and dfs iterators.
9 /// This file contains bfs and dfs iterator classes.
11 // /// \author Marton Makai
17 #include <hugo/invalid.h>
21 /// Bfs searches for the nodes wich are not marked in
23 /// Reached have to work as read-write bool Node-map.
25 template <typename Graph, /*typename OutEdgeIt,*/
26 typename ReachedMap/*=typename Graph::NodeMap<bool>*/ >
29 typedef typename Graph::Node Node;
30 typedef typename Graph::OutEdgeIt OutEdgeIt;
32 std::queue<Node> bfs_queue;
34 bool b_node_newly_reached;
35 OutEdgeIt actual_edge;
38 /// In that constructor \c _reached have to be a reference
39 /// for a bool Node-map. The algorithm will search in a bfs order for
40 /// the nodes which are \c false initially
41 BfsIterator(const Graph& _graph, ReachedMap& _reached) :
42 graph(&_graph), reached(_reached),
43 own_reached_map(false) { }
44 /// The same as above, but the map storing the reached nodes
45 /// is constructed dynamically to everywhere false.
46 BfsIterator(const Graph& _graph) :
47 graph(&_graph), reached(*(new ReachedMap(*graph /*, false*/))),
48 own_reached_map(true) { }
49 /// The map storing the reached nodes have to be destroyed if
50 /// it was constructed dynamically
51 ~BfsIterator() { if (own_reached_map) delete &reached; }
52 /// This method markes \c s reached.
53 /// If the queue is empty, then \c s is pushed in the bfs queue
54 /// and the first out-edge is processed.
55 /// If the queue is not empty, then \c s is simply pushed.
56 void pushAndSetReached(Node s) {
58 if (bfs_queue.empty()) {
60 graph->first(actual_edge, s);
61 if (graph->valid(actual_edge)) {
62 Node w=graph->bNode(actual_edge);
66 b_node_newly_reached=true;
68 b_node_newly_reached=false;
75 /// As \c BfsIterator<Graph, ReachedMap> works as an edge-iterator,
76 /// its \c operator++() iterates on the edges in a bfs order.
77 BfsIterator<Graph, /*OutEdgeIt,*/ ReachedMap>&
79 if (graph->valid(actual_edge)) {
80 graph->next(actual_edge);
81 if (graph->valid(actual_edge)) {
82 Node w=graph->bNode(actual_edge);
86 b_node_newly_reached=true;
88 b_node_newly_reached=false;
93 if (!bfs_queue.empty()) {
94 graph->first(actual_edge, bfs_queue.front());
95 if (graph->valid(actual_edge)) {
96 Node w=graph->bNode(actual_edge);
100 b_node_newly_reached=true;
102 b_node_newly_reached=false;
109 /// Returns true iff the algorithm is finished.
110 bool finished() const { return bfs_queue.empty(); }
111 /// The conversion operator makes for converting the bfs-iterator
112 /// to an \c out-edge-iterator.
113 ///\bug Edge have to be in HUGO 0.2
114 operator OutEdgeIt() const { return actual_edge; }
115 /// Returns if b-node has been reached just now.
116 bool isBNodeNewlyReached() const { return b_node_newly_reached; }
117 /// Returns if a-node is examined.
118 bool isANodeExamined() const { return !(graph->valid(actual_edge)); }
119 /// Returns a-node of the actual edge, so does if the edge is invalid.
120 Node aNode() const { return bfs_queue.front(); }
121 /// \pre The actual edge have to be valid.
122 Node bNode() const { return graph->bNode(actual_edge); }
124 const ReachedMap& getReachedMap() const { return reached; }
126 const std::queue<Node>& getBfsQueue() const { return bfs_queue; }
129 /// Bfs searches for the nodes wich are not marked in
131 /// Reached have to work as a read-write bool Node-map,
132 /// Pred is a write Edge Node-map and
133 /// Dist is a read-write Node-map of integral value, have to be.
135 template <typename Graph,
136 typename ReachedMap=typename Graph::template NodeMap<bool>,
138 =typename Graph::template NodeMap<typename Graph::Edge>,
139 typename DistMap=typename Graph::template NodeMap<int> >
140 class Bfs : public BfsIterator<Graph, ReachedMap> {
141 typedef BfsIterator<Graph, ReachedMap> Parent;
143 typedef typename Parent::Node Node;
147 /// The algorithm will search in a bfs order for
148 /// the nodes which are \c false initially.
149 /// The constructor makes no initial changes on the maps.
150 Bfs<Graph, ReachedMap, PredMap, DistMap>(const Graph& _graph, ReachedMap& _reached, PredMap& _pred, DistMap& _dist) : BfsIterator<Graph, ReachedMap>(_graph, _reached), pred(&_pred), dist(&_dist) { }
151 /// \c s is marked to be reached and pushed in the bfs queue.
152 /// If the queue is empty, then the first out-edge is processed.
153 /// If \c s was not marked previously, then
154 /// in addition its pred is set to be \c INVALID, and dist to \c 0.
155 /// if \c s was marked previuosly, then it is simply pushed.
157 if (this->reached[s]) {
158 Parent::pushAndSetReached(s);
160 Parent::pushAndSetReached(s);
161 pred.set(s, INVALID);
165 /// A bfs is processed from \c s.
168 while (!this->finished()) this->operator++();
170 /// Beside the bfs iteration, \c pred and \dist are saved in a
171 /// newly reached node.
172 Bfs<Graph, ReachedMap, PredMap, DistMap>& operator++() {
173 Parent::operator++();
174 if (this->graph->valid(this->actual_edge) && this->b_node_newly_reached)
176 pred.set(this->bNode(), this->actual_edge);
177 dist.set(this->bNode(), dist[this->aNode()]);
182 const PredMap& getPredMap() const { return pred; }
184 const DistMap& getDistMap() const { return dist; }
187 /// Dfs searches for the nodes wich are not marked in
189 /// Reached have to be a read-write bool Node-map.
191 template <typename Graph, /*typename OutEdgeIt,*/
192 typename ReachedMap/*=typename Graph::NodeMap<bool>*/ >
195 typedef typename Graph::Node Node;
196 typedef typename Graph::OutEdgeIt OutEdgeIt;
198 std::stack<OutEdgeIt> dfs_stack;
199 bool b_node_newly_reached;
200 OutEdgeIt actual_edge;
203 bool own_reached_map;
205 /// In that constructor \c _reached have to be a reference
206 /// for a bool Node-map. The algorithm will search in a dfs order for
207 /// the nodes which are \c false initially
208 DfsIterator(const Graph& _graph, ReachedMap& _reached) :
209 graph(&_graph), reached(_reached),
210 own_reached_map(false) { }
211 /// The same as above, but the map of reached nodes is
212 /// constructed dynamically
213 /// to everywhere false.
214 DfsIterator(const Graph& _graph) :
215 graph(&_graph), reached(*(new ReachedMap(*graph /*, false*/))),
216 own_reached_map(true) { }
217 ~DfsIterator() { if (own_reached_map) delete &reached; }
218 /// This method markes s reached and first out-edge is processed.
219 void pushAndSetReached(Node s) {
221 reached.set(s, true);
226 /// As \c DfsIterator<Graph, ReachedMap> works as an edge-iterator,
227 /// its \c operator++() iterates on the edges in a dfs order.
228 DfsIterator<Graph, /*OutEdgeIt,*/ ReachedMap>&
230 actual_edge=dfs_stack.top();
231 //actual_node=G.aNode(actual_edge);
232 if (graph->valid(actual_edge)/*.valid()*/) {
233 Node w=graph->bNode(actual_edge);
239 reached.set(w, true);
240 b_node_newly_reached=true;
242 actual_node=graph->aNode(actual_edge);
243 graph->next(dfs_stack.top());
244 b_node_newly_reached=false;
247 //actual_node=G.aNode(dfs_stack.top());
252 /// Returns true iff the algorithm is finished.
253 bool finished() const { return dfs_stack.empty(); }
254 /// The conversion operator makes for converting the bfs-iterator
255 /// to an \c out-edge-iterator.
256 ///\bug Edge have to be in HUGO 0.2
257 operator OutEdgeIt() const { return actual_edge; }
258 /// Returns if b-node has been reached just now.
259 bool isBNodeNewlyReached() const { return b_node_newly_reached; }
260 /// Returns if a-node is examined.
261 bool isANodeExamined() const { return !(graph->valid(actual_edge)); }
262 /// Returns a-node of the actual edge, so does if the edge is invalid.
263 Node aNode() const { return actual_node; /*FIXME*/}
264 /// Returns b-node of the actual edge.
265 /// \pre The actual edge have to be valid.
266 Node bNode() const { return graph->bNode(actual_edge); }
268 const ReachedMap& getReachedMap() const { return reached; }
270 const std::stack<OutEdgeIt>& getDfsStack() const { return dfs_stack; }
273 /// Dfs searches for the nodes wich are not marked in
275 /// Reached is a read-write bool Node-map,
276 /// Pred is a write Node-map, have to be.
278 template <typename Graph,
279 typename ReachedMap=typename Graph::template NodeMap<bool>,
281 =typename Graph::template NodeMap<typename Graph::Edge> >
282 class Dfs : public DfsIterator<Graph, ReachedMap> {
283 typedef DfsIterator<Graph, ReachedMap> Parent;
285 typedef typename Parent::Node Node;
288 /// The algorithm will search in a dfs order for
289 /// the nodes which are \c false initially.
290 /// The constructor makes no initial changes on the maps.
291 Dfs<Graph, ReachedMap, PredMap>(const Graph& _graph, ReachedMap& _reached, PredMap& _pred) : DfsIterator<Graph, ReachedMap>(_graph, _reached), pred(&_pred) { }
292 /// \c s is marked to be reached and pushed in the bfs queue.
293 /// If the queue is empty, then the first out-edge is processed.
294 /// If \c s was not marked previously, then
295 /// in addition its pred is set to be \c INVALID.
296 /// if \c s was marked previuosly, then it is simply pushed.
298 if (this->reached[s]) {
299 Parent::pushAndSetReached(s);
301 Parent::pushAndSetReached(s);
302 pred.set(s, INVALID);
305 /// A bfs is processed from \c s.
308 while (!this->finished()) this->operator++();
310 /// Beside the dfs iteration, \c pred is saved in a
311 /// newly reached node.
312 Dfs<Graph, ReachedMap, PredMap>& operator++() {
313 Parent::operator++();
314 if (this->graph->valid(this->actual_edge) && this->b_node_newly_reached)
316 pred.set(this->bNode(), this->actual_edge);
321 const PredMap& getPredMap() const { return pred; }
327 #endif //HUGO_BFS_DFS_H