breadth_first_search.hpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. //
  2. //=======================================================================
  3. // Copyright 1997, 1998, 1999, 2000 University of Notre Dame.
  4. // Authors: Andrew Lumsdaine, Lie-Quan Lee, Jeremy G. Siek
  5. //
  6. // Distributed under the Boost Software License, Version 1.0. (See
  7. // accompanying file LICENSE_1_0.txt or copy at
  8. // http://www.boost.org/LICENSE_1_0.txt)
  9. //=======================================================================
  10. //
  11. #ifndef BOOST_GRAPH_BREADTH_FIRST_SEARCH_HPP
  12. #define BOOST_GRAPH_BREADTH_FIRST_SEARCH_HPP
  13. /*
  14. Breadth First Search Algorithm (Cormen, Leiserson, and Rivest p. 470)
  15. */
  16. #include <boost/config.hpp>
  17. #include <vector>
  18. #include <boost/pending/queue.hpp>
  19. #include <boost/graph/graph_traits.hpp>
  20. #include <boost/graph/graph_concepts.hpp>
  21. #include <boost/graph/visitors.hpp>
  22. #include <boost/graph/named_function_params.hpp>
  23. #include <boost/graph/overloading.hpp>
  24. #include <boost/graph/graph_concepts.hpp>
  25. #include <boost/graph/two_bit_color_map.hpp>
  26. #include <boost/concept/assert.hpp>
  27. #ifdef BOOST_GRAPH_USE_MPI
  28. #include <boost/graph/distributed/concepts.hpp>
  29. #endif // BOOST_GRAPH_USE_MPI
  30. namespace boost {
  31. template <class Visitor, class Graph>
  32. struct BFSVisitorConcept {
  33. void constraints() {
  34. BOOST_CONCEPT_ASSERT(( CopyConstructibleConcept<Visitor> ));
  35. vis.initialize_vertex(u, g);
  36. vis.discover_vertex(u, g);
  37. vis.examine_vertex(u, g);
  38. vis.examine_edge(e, g);
  39. vis.tree_edge(e, g);
  40. vis.non_tree_edge(e, g);
  41. vis.gray_target(e, g);
  42. vis.black_target(e, g);
  43. vis.finish_vertex(u, g);
  44. }
  45. Visitor vis;
  46. Graph g;
  47. typename graph_traits<Graph>::vertex_descriptor u;
  48. typename graph_traits<Graph>::edge_descriptor e;
  49. };
  50. // Multiple-source version
  51. template <class IncidenceGraph, class Buffer, class BFSVisitor,
  52. class ColorMap, class SourceIterator>
  53. void breadth_first_visit
  54. (const IncidenceGraph& g,
  55. SourceIterator sources_begin, SourceIterator sources_end,
  56. Buffer& Q, BFSVisitor vis, ColorMap color)
  57. {
  58. BOOST_CONCEPT_ASSERT(( IncidenceGraphConcept<IncidenceGraph> ));
  59. typedef graph_traits<IncidenceGraph> GTraits;
  60. typedef typename GTraits::vertex_descriptor Vertex;
  61. BOOST_CONCEPT_ASSERT(( BFSVisitorConcept<BFSVisitor, IncidenceGraph> ));
  62. BOOST_CONCEPT_ASSERT(( ReadWritePropertyMapConcept<ColorMap, Vertex> ));
  63. typedef typename property_traits<ColorMap>::value_type ColorValue;
  64. typedef color_traits<ColorValue> Color;
  65. typename GTraits::out_edge_iterator ei, ei_end;
  66. for (; sources_begin != sources_end; ++sources_begin) {
  67. Vertex s = *sources_begin;
  68. put(color, s, Color::gray()); vis.discover_vertex(s, g);
  69. Q.push(s);
  70. }
  71. while (! Q.empty()) {
  72. Vertex u = Q.top(); Q.pop(); vis.examine_vertex(u, g);
  73. for (boost::tie(ei, ei_end) = out_edges(u, g); ei != ei_end; ++ei) {
  74. Vertex v = target(*ei, g); vis.examine_edge(*ei, g);
  75. ColorValue v_color = get(color, v);
  76. if (v_color == Color::white()) { vis.tree_edge(*ei, g);
  77. put(color, v, Color::gray()); vis.discover_vertex(v, g);
  78. Q.push(v);
  79. } else { vis.non_tree_edge(*ei, g);
  80. if (v_color == Color::gray()) vis.gray_target(*ei, g);
  81. else vis.black_target(*ei, g);
  82. }
  83. } // end for
  84. put(color, u, Color::black()); vis.finish_vertex(u, g);
  85. } // end while
  86. } // breadth_first_visit
  87. // Single-source version
  88. template <class IncidenceGraph, class Buffer, class BFSVisitor,
  89. class ColorMap>
  90. void breadth_first_visit
  91. (const IncidenceGraph& g,
  92. typename graph_traits<IncidenceGraph>::vertex_descriptor s,
  93. Buffer& Q, BFSVisitor vis, ColorMap color)
  94. {
  95. typename graph_traits<IncidenceGraph>::vertex_descriptor sources[1] = {s};
  96. breadth_first_visit(g, sources, sources + 1, Q, vis, color);
  97. }
  98. template <class VertexListGraph, class SourceIterator,
  99. class Buffer, class BFSVisitor,
  100. class ColorMap>
  101. void breadth_first_search
  102. (const VertexListGraph& g,
  103. SourceIterator sources_begin, SourceIterator sources_end,
  104. Buffer& Q, BFSVisitor vis, ColorMap color)
  105. {
  106. // Initialization
  107. typedef typename property_traits<ColorMap>::value_type ColorValue;
  108. typedef color_traits<ColorValue> Color;
  109. typename boost::graph_traits<VertexListGraph>::vertex_iterator i, i_end;
  110. for (boost::tie(i, i_end) = vertices(g); i != i_end; ++i) {
  111. vis.initialize_vertex(*i, g);
  112. put(color, *i, Color::white());
  113. }
  114. breadth_first_visit(g, sources_begin, sources_end, Q, vis, color);
  115. }
  116. template <class VertexListGraph, class Buffer, class BFSVisitor,
  117. class ColorMap>
  118. void breadth_first_search
  119. (const VertexListGraph& g,
  120. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  121. Buffer& Q, BFSVisitor vis, ColorMap color)
  122. {
  123. typename graph_traits<VertexListGraph>::vertex_descriptor sources[1] = {s};
  124. breadth_first_search(g, sources, sources + 1, Q, vis, color);
  125. }
  126. namespace graph { struct bfs_visitor_event_not_overridden {}; }
  127. template <class Visitors = null_visitor>
  128. class bfs_visitor {
  129. public:
  130. bfs_visitor() { }
  131. bfs_visitor(Visitors vis) : m_vis(vis) { }
  132. template <class Vertex, class Graph>
  133. graph::bfs_visitor_event_not_overridden
  134. initialize_vertex(Vertex u, Graph& g)
  135. {
  136. invoke_visitors(m_vis, u, g, ::boost::on_initialize_vertex());
  137. return graph::bfs_visitor_event_not_overridden();
  138. }
  139. template <class Vertex, class Graph>
  140. graph::bfs_visitor_event_not_overridden
  141. discover_vertex(Vertex u, Graph& g)
  142. {
  143. invoke_visitors(m_vis, u, g, ::boost::on_discover_vertex());
  144. return graph::bfs_visitor_event_not_overridden();
  145. }
  146. template <class Vertex, class Graph>
  147. graph::bfs_visitor_event_not_overridden
  148. examine_vertex(Vertex u, Graph& g)
  149. {
  150. invoke_visitors(m_vis, u, g, ::boost::on_examine_vertex());
  151. return graph::bfs_visitor_event_not_overridden();
  152. }
  153. template <class Edge, class Graph>
  154. graph::bfs_visitor_event_not_overridden
  155. examine_edge(Edge e, Graph& g)
  156. {
  157. invoke_visitors(m_vis, e, g, ::boost::on_examine_edge());
  158. return graph::bfs_visitor_event_not_overridden();
  159. }
  160. template <class Edge, class Graph>
  161. graph::bfs_visitor_event_not_overridden
  162. tree_edge(Edge e, Graph& g)
  163. {
  164. invoke_visitors(m_vis, e, g, ::boost::on_tree_edge());
  165. return graph::bfs_visitor_event_not_overridden();
  166. }
  167. template <class Edge, class Graph>
  168. graph::bfs_visitor_event_not_overridden
  169. non_tree_edge(Edge e, Graph& g)
  170. {
  171. invoke_visitors(m_vis, e, g, ::boost::on_non_tree_edge());
  172. return graph::bfs_visitor_event_not_overridden();
  173. }
  174. template <class Edge, class Graph>
  175. graph::bfs_visitor_event_not_overridden
  176. gray_target(Edge e, Graph& g)
  177. {
  178. invoke_visitors(m_vis, e, g, ::boost::on_gray_target());
  179. return graph::bfs_visitor_event_not_overridden();
  180. }
  181. template <class Edge, class Graph>
  182. graph::bfs_visitor_event_not_overridden
  183. black_target(Edge e, Graph& g)
  184. {
  185. invoke_visitors(m_vis, e, g, ::boost::on_black_target());
  186. return graph::bfs_visitor_event_not_overridden();
  187. }
  188. template <class Vertex, class Graph>
  189. graph::bfs_visitor_event_not_overridden
  190. finish_vertex(Vertex u, Graph& g)
  191. {
  192. invoke_visitors(m_vis, u, g, ::boost::on_finish_vertex());
  193. return graph::bfs_visitor_event_not_overridden();
  194. }
  195. BOOST_GRAPH_EVENT_STUB(on_initialize_vertex,bfs)
  196. BOOST_GRAPH_EVENT_STUB(on_discover_vertex,bfs)
  197. BOOST_GRAPH_EVENT_STUB(on_examine_vertex,bfs)
  198. BOOST_GRAPH_EVENT_STUB(on_examine_edge,bfs)
  199. BOOST_GRAPH_EVENT_STUB(on_tree_edge,bfs)
  200. BOOST_GRAPH_EVENT_STUB(on_non_tree_edge,bfs)
  201. BOOST_GRAPH_EVENT_STUB(on_gray_target,bfs)
  202. BOOST_GRAPH_EVENT_STUB(on_black_target,bfs)
  203. BOOST_GRAPH_EVENT_STUB(on_finish_vertex,bfs)
  204. protected:
  205. Visitors m_vis;
  206. };
  207. template <class Visitors>
  208. bfs_visitor<Visitors>
  209. make_bfs_visitor(Visitors vis) {
  210. return bfs_visitor<Visitors>(vis);
  211. }
  212. typedef bfs_visitor<> default_bfs_visitor;
  213. namespace detail {
  214. template <class VertexListGraph, class ColorMap, class BFSVisitor,
  215. class P, class T, class R>
  216. void bfs_helper
  217. (VertexListGraph& g,
  218. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  219. ColorMap color,
  220. BFSVisitor vis,
  221. const bgl_named_params<P, T, R>& params,
  222. boost::mpl::false_)
  223. {
  224. typedef graph_traits<VertexListGraph> Traits;
  225. // Buffer default
  226. typedef typename Traits::vertex_descriptor Vertex;
  227. typedef boost::queue<Vertex> queue_t;
  228. queue_t Q;
  229. breadth_first_search
  230. (g, s,
  231. choose_param(get_param(params, buffer_param_t()), boost::ref(Q)).get(),
  232. vis, color);
  233. }
  234. #ifdef BOOST_GRAPH_USE_MPI
  235. template <class DistributedGraph, class ColorMap, class BFSVisitor,
  236. class P, class T, class R>
  237. void bfs_helper
  238. (DistributedGraph& g,
  239. typename graph_traits<DistributedGraph>::vertex_descriptor s,
  240. ColorMap color,
  241. BFSVisitor vis,
  242. const bgl_named_params<P, T, R>& params,
  243. boost::mpl::true_);
  244. #endif // BOOST_GRAPH_USE_MPI
  245. //-------------------------------------------------------------------------
  246. // Choose between default color and color parameters. Using
  247. // function dispatching so that we don't require vertex index if
  248. // the color default is not being used.
  249. template <class ColorMap>
  250. struct bfs_dispatch {
  251. template <class VertexListGraph, class P, class T, class R>
  252. static void apply
  253. (VertexListGraph& g,
  254. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  255. const bgl_named_params<P, T, R>& params,
  256. ColorMap color)
  257. {
  258. bfs_helper
  259. (g, s, color,
  260. choose_param(get_param(params, graph_visitor),
  261. make_bfs_visitor(null_visitor())),
  262. params,
  263. boost::mpl::bool_<
  264. boost::is_base_and_derived<
  265. distributed_graph_tag,
  266. typename graph_traits<VertexListGraph>::traversal_category>::value>());
  267. }
  268. };
  269. template <>
  270. struct bfs_dispatch<param_not_found> {
  271. template <class VertexListGraph, class P, class T, class R>
  272. static void apply
  273. (VertexListGraph& g,
  274. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  275. const bgl_named_params<P, T, R>& params,
  276. param_not_found)
  277. {
  278. null_visitor null_vis;
  279. bfs_helper
  280. (g, s,
  281. make_two_bit_color_map
  282. (num_vertices(g),
  283. choose_const_pmap(get_param(params, vertex_index),
  284. g, vertex_index)),
  285. choose_param(get_param(params, graph_visitor),
  286. make_bfs_visitor(null_vis)),
  287. params,
  288. boost::mpl::bool_<
  289. boost::is_base_and_derived<
  290. distributed_graph_tag,
  291. typename graph_traits<VertexListGraph>::traversal_category>::value>());
  292. }
  293. };
  294. } // namespace detail
  295. #if 1
  296. // Named Parameter Variant
  297. template <class VertexListGraph, class P, class T, class R>
  298. void breadth_first_search
  299. (const VertexListGraph& g,
  300. typename graph_traits<VertexListGraph>::vertex_descriptor s,
  301. const bgl_named_params<P, T, R>& params)
  302. {
  303. // The graph is passed by *const* reference so that graph adaptors
  304. // (temporaries) can be passed into this function. However, the
  305. // graph is not really const since we may write to property maps
  306. // of the graph.
  307. VertexListGraph& ng = const_cast<VertexListGraph&>(g);
  308. typedef typename get_param_type< vertex_color_t, bgl_named_params<P,T,R> >::type C;
  309. detail::bfs_dispatch<C>::apply(ng, s, params,
  310. get_param(params, vertex_color));
  311. }
  312. #endif
  313. // This version does not initialize colors, user has to.
  314. template <class IncidenceGraph, class P, class T, class R>
  315. void breadth_first_visit
  316. (const IncidenceGraph& g,
  317. typename graph_traits<IncidenceGraph>::vertex_descriptor s,
  318. const bgl_named_params<P, T, R>& params)
  319. {
  320. // The graph is passed by *const* reference so that graph adaptors
  321. // (temporaries) can be passed into this function. However, the
  322. // graph is not really const since we may write to property maps
  323. // of the graph.
  324. IncidenceGraph& ng = const_cast<IncidenceGraph&>(g);
  325. typedef graph_traits<IncidenceGraph> Traits;
  326. // Buffer default
  327. typedef typename Traits::vertex_descriptor vertex_descriptor;
  328. typedef boost::queue<vertex_descriptor> queue_t;
  329. queue_t Q;
  330. breadth_first_visit
  331. (ng, s,
  332. choose_param(get_param(params, buffer_param_t()), boost::ref(Q)).get(),
  333. choose_param(get_param(params, graph_visitor),
  334. make_bfs_visitor(null_visitor())),
  335. choose_pmap(get_param(params, vertex_color), ng, vertex_color)
  336. );
  337. }
  338. namespace graph {
  339. namespace detail {
  340. template <typename Graph, typename Source>
  341. struct breadth_first_search_impl {
  342. typedef void result_type;
  343. template <typename ArgPack>
  344. void operator()(const Graph& g, const Source& source, const ArgPack& arg_pack) {
  345. using namespace boost::graph::keywords;
  346. typename boost::graph_traits<Graph>::vertex_descriptor sources[1] = {source};
  347. boost::queue<typename boost::graph_traits<Graph>::vertex_descriptor> Q;
  348. boost::breadth_first_search(g,
  349. &sources[0],
  350. &sources[1],
  351. boost::unwrap_ref(arg_pack[_buffer | boost::ref(Q)]),
  352. arg_pack[_visitor | make_bfs_visitor(null_visitor())],
  353. boost::detail::make_color_map_from_arg_pack(g, arg_pack));
  354. }
  355. };
  356. }
  357. BOOST_GRAPH_MAKE_FORWARDING_FUNCTION(breadth_first_search, 2, 4)
  358. }
  359. #if 0
  360. // Named Parameter Variant
  361. BOOST_GRAPH_MAKE_OLD_STYLE_PARAMETER_FUNCTION(breadth_first_search, 2)
  362. #endif
  363. } // namespace boost
  364. #ifdef BOOST_GRAPH_USE_MPI
  365. # include <boost/graph/distributed/breadth_first_search.hpp>
  366. #endif
  367. #endif // BOOST_GRAPH_BREADTH_FIRST_SEARCH_HPP