strong_components.hpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  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_STRONG_COMPONENTS_HPP
  12. #define BOOST_GRAPH_STRONG_COMPONENTS_HPP
  13. #include <stack>
  14. #include <boost/config.hpp>
  15. #include <boost/graph/depth_first_search.hpp>
  16. #include <boost/type_traits/conversion_traits.hpp>
  17. #include <boost/static_assert.hpp>
  18. #include <boost/graph/overloading.hpp>
  19. #include <boost/concept/assert.hpp>
  20. namespace boost {
  21. //==========================================================================
  22. // This is Tarjan's algorithm for strongly connected components
  23. // from his paper "Depth first search and linear graph algorithms".
  24. // It calculates the components in a single application of DFS.
  25. // We implement the algorithm as a dfs-visitor.
  26. namespace detail {
  27. template <typename ComponentMap, typename RootMap, typename DiscoverTime,
  28. typename Stack>
  29. class tarjan_scc_visitor : public dfs_visitor<>
  30. {
  31. typedef typename property_traits<ComponentMap>::value_type comp_type;
  32. typedef typename property_traits<DiscoverTime>::value_type time_type;
  33. public:
  34. tarjan_scc_visitor(ComponentMap comp_map, RootMap r, DiscoverTime d,
  35. comp_type& c_, Stack& s_)
  36. : c(c_), comp(comp_map), root(r), discover_time(d),
  37. dfs_time(time_type()), s(s_) { }
  38. template <typename Graph>
  39. void discover_vertex(typename graph_traits<Graph>::vertex_descriptor v,
  40. const Graph&) {
  41. put(root, v, v);
  42. put(comp, v, (std::numeric_limits<comp_type>::max)());
  43. put(discover_time, v, dfs_time++);
  44. s.push(v);
  45. }
  46. template <typename Graph>
  47. void finish_vertex(typename graph_traits<Graph>::vertex_descriptor v,
  48. const Graph& g) {
  49. typename graph_traits<Graph>::vertex_descriptor w;
  50. typename graph_traits<Graph>::out_edge_iterator ei, ei_end;
  51. for (boost::tie(ei, ei_end) = out_edges(v, g); ei != ei_end; ++ei) {
  52. w = target(*ei, g);
  53. if (get(comp, w) == (std::numeric_limits<comp_type>::max)())
  54. put(root, v, this->min_discover_time(get(root,v), get(root,w)));
  55. }
  56. if (get(root, v) == v) {
  57. do {
  58. w = s.top(); s.pop();
  59. put(comp, w, c);
  60. } while (w != v);
  61. ++c;
  62. }
  63. }
  64. private:
  65. template <typename Vertex>
  66. Vertex min_discover_time(Vertex u, Vertex v) {
  67. return get(discover_time, u) < get(discover_time,v) ? u : v;
  68. }
  69. comp_type& c;
  70. ComponentMap comp;
  71. RootMap root;
  72. DiscoverTime discover_time;
  73. time_type dfs_time;
  74. Stack& s;
  75. };
  76. template <class Graph, class ComponentMap, class RootMap,
  77. class DiscoverTime, class P, class T, class R>
  78. typename property_traits<ComponentMap>::value_type
  79. strong_components_impl
  80. (const Graph& g, // Input
  81. ComponentMap comp, // Output
  82. // Internal record keeping
  83. RootMap root,
  84. DiscoverTime discover_time,
  85. const bgl_named_params<P, T, R>& params)
  86. {
  87. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  88. BOOST_CONCEPT_ASSERT(( ReadWritePropertyMapConcept<ComponentMap, Vertex> ));
  89. BOOST_CONCEPT_ASSERT(( ReadWritePropertyMapConcept<RootMap, Vertex> ));
  90. typedef typename property_traits<RootMap>::value_type RootV;
  91. BOOST_CONCEPT_ASSERT(( ConvertibleConcept<RootV, Vertex> ));
  92. BOOST_CONCEPT_ASSERT(( ReadWritePropertyMapConcept<DiscoverTime, Vertex> ));
  93. typename property_traits<ComponentMap>::value_type total = 0;
  94. std::stack<Vertex> s;
  95. detail::tarjan_scc_visitor<ComponentMap, RootMap, DiscoverTime,
  96. std::stack<Vertex> >
  97. vis(comp, root, discover_time, total, s);
  98. depth_first_search(g, params.visitor(vis));
  99. return total;
  100. }
  101. //-------------------------------------------------------------------------
  102. // The dispatch functions handle the defaults for the rank and discover
  103. // time property maps.
  104. // dispatch with class specialization to avoid VC++ bug
  105. template <class DiscoverTimeMap>
  106. struct strong_comp_dispatch2 {
  107. template <class Graph, class ComponentMap, class RootMap, class P, class T, class R>
  108. inline static typename property_traits<ComponentMap>::value_type
  109. apply(const Graph& g,
  110. ComponentMap comp,
  111. RootMap r_map,
  112. const bgl_named_params<P, T, R>& params,
  113. DiscoverTimeMap time_map)
  114. {
  115. return strong_components_impl(g, comp, r_map, time_map, params);
  116. }
  117. };
  118. template <>
  119. struct strong_comp_dispatch2<param_not_found> {
  120. template <class Graph, class ComponentMap, class RootMap,
  121. class P, class T, class R>
  122. inline static typename property_traits<ComponentMap>::value_type
  123. apply(const Graph& g,
  124. ComponentMap comp,
  125. RootMap r_map,
  126. const bgl_named_params<P, T, R>& params,
  127. param_not_found)
  128. {
  129. typedef typename graph_traits<Graph>::vertices_size_type size_type;
  130. size_type n = num_vertices(g) > 0 ? num_vertices(g) : 1;
  131. std::vector<size_type> time_vec(n);
  132. return strong_components_impl
  133. (g, comp, r_map,
  134. make_iterator_property_map(time_vec.begin(), choose_const_pmap
  135. (get_param(params, vertex_index),
  136. g, vertex_index), time_vec[0]),
  137. params);
  138. }
  139. };
  140. template <class Graph, class ComponentMap, class RootMap,
  141. class P, class T, class R, class DiscoverTimeMap>
  142. inline typename property_traits<ComponentMap>::value_type
  143. scc_helper2(const Graph& g,
  144. ComponentMap comp,
  145. RootMap r_map,
  146. const bgl_named_params<P, T, R>& params,
  147. DiscoverTimeMap time_map)
  148. {
  149. return strong_comp_dispatch2<DiscoverTimeMap>::apply(g, comp, r_map, params, time_map);
  150. }
  151. template <class RootMap>
  152. struct strong_comp_dispatch1 {
  153. template <class Graph, class ComponentMap, class P, class T, class R>
  154. inline static typename property_traits<ComponentMap>::value_type
  155. apply(const Graph& g,
  156. ComponentMap comp,
  157. const bgl_named_params<P, T, R>& params,
  158. RootMap r_map)
  159. {
  160. return scc_helper2(g, comp, r_map, params, get_param(params, vertex_discover_time));
  161. }
  162. };
  163. template <>
  164. struct strong_comp_dispatch1<param_not_found> {
  165. template <class Graph, class ComponentMap,
  166. class P, class T, class R>
  167. inline static typename property_traits<ComponentMap>::value_type
  168. apply(const Graph& g,
  169. ComponentMap comp,
  170. const bgl_named_params<P, T, R>& params,
  171. param_not_found)
  172. {
  173. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  174. typename std::vector<Vertex>::size_type
  175. n = num_vertices(g) > 0 ? num_vertices(g) : 1;
  176. std::vector<Vertex> root_vec(n);
  177. return scc_helper2
  178. (g, comp,
  179. make_iterator_property_map(root_vec.begin(), choose_const_pmap
  180. (get_param(params, vertex_index),
  181. g, vertex_index), root_vec[0]),
  182. params,
  183. get_param(params, vertex_discover_time));
  184. }
  185. };
  186. template <class Graph, class ComponentMap, class RootMap,
  187. class P, class T, class R>
  188. inline typename property_traits<ComponentMap>::value_type
  189. scc_helper1(const Graph& g,
  190. ComponentMap comp,
  191. const bgl_named_params<P, T, R>& params,
  192. RootMap r_map)
  193. {
  194. return detail::strong_comp_dispatch1<RootMap>::apply(g, comp, params,
  195. r_map);
  196. }
  197. } // namespace detail
  198. template <class Graph, class ComponentMap,
  199. class P, class T, class R>
  200. inline typename property_traits<ComponentMap>::value_type
  201. strong_components(const Graph& g, ComponentMap comp,
  202. const bgl_named_params<P, T, R>& params
  203. BOOST_GRAPH_ENABLE_IF_MODELS_PARM(Graph, vertex_list_graph_tag))
  204. {
  205. typedef typename graph_traits<Graph>::directed_category DirCat;
  206. BOOST_STATIC_ASSERT((is_convertible<DirCat*, directed_tag*>::value == true));
  207. return detail::scc_helper1(g, comp, params,
  208. get_param(params, vertex_root_t()));
  209. }
  210. template <class Graph, class ComponentMap>
  211. inline typename property_traits<ComponentMap>::value_type
  212. strong_components(const Graph& g, ComponentMap comp
  213. BOOST_GRAPH_ENABLE_IF_MODELS_PARM(Graph, vertex_list_graph_tag))
  214. {
  215. typedef typename graph_traits<Graph>::directed_category DirCat;
  216. BOOST_STATIC_ASSERT((is_convertible<DirCat*, directed_tag*>::value == true));
  217. bgl_named_params<int, int> params(0);
  218. return strong_components(g, comp, params);
  219. }
  220. template <typename Graph, typename ComponentMap, typename ComponentLists>
  221. void build_component_lists
  222. (const Graph& g,
  223. typename graph_traits<Graph>::vertices_size_type num_scc,
  224. ComponentMap component_number,
  225. ComponentLists& components)
  226. {
  227. components.resize(num_scc);
  228. typename graph_traits<Graph>::vertex_iterator vi, vi_end;
  229. for (boost::tie(vi, vi_end) = vertices(g); vi != vi_end; ++vi)
  230. components[component_number[*vi]].push_back(*vi);
  231. }
  232. } // namespace boost
  233. #include <queue>
  234. #include <vector>
  235. #include <boost/graph/transpose_graph.hpp>
  236. #include <boost/pending/indirect_cmp.hpp>
  237. #include <boost/graph/connected_components.hpp> // for components_recorder
  238. namespace boost {
  239. //==========================================================================
  240. // This is the version of strongly connected components from
  241. // "Intro. to Algorithms" by Cormen, Leiserson, Rivest, which was
  242. // adapted from "Data Structure and Algorithms" by Aho, Hopcroft,
  243. // and Ullman, who credit the algorithm to S.R. Kosaraju and M. Sharir.
  244. // The algorithm is based on computing DFS forests the graph
  245. // and its transpose.
  246. // This algorithm is slower than Tarjan's by a constant factor, uses
  247. // more memory, and puts more requirements on the graph type.
  248. template <class Graph, class DFSVisitor, class ComponentsMap,
  249. class DiscoverTime, class FinishTime,
  250. class ColorMap>
  251. typename property_traits<ComponentsMap>::value_type
  252. kosaraju_strong_components(Graph& G, ComponentsMap c,
  253. FinishTime finish_time, ColorMap color)
  254. {
  255. BOOST_CONCEPT_ASSERT(( MutableGraphConcept<Graph> ));
  256. // ...
  257. typedef typename graph_traits<Graph>::vertex_descriptor Vertex;
  258. typedef typename property_traits<ColorMap>::value_type ColorValue;
  259. typedef color_traits<ColorValue> Color;
  260. typename property_traits<FinishTime>::value_type time = 0;
  261. depth_first_search
  262. (G, make_dfs_visitor(stamp_times(finish_time, time, on_finish_vertex())),
  263. color);
  264. Graph G_T(num_vertices(G));
  265. transpose_graph(G, G_T);
  266. typedef typename property_traits<ComponentsMap>::value_type count_type;
  267. count_type c_count(0);
  268. detail::components_recorder<ComponentsMap>
  269. vis(c, c_count);
  270. // initialize G_T
  271. typename graph_traits<Graph>::vertex_iterator ui, ui_end;
  272. for (boost::tie(ui, ui_end) = vertices(G_T); ui != ui_end; ++ui)
  273. put(color, *ui, Color::white());
  274. typedef typename property_traits<FinishTime>::value_type D;
  275. typedef indirect_cmp< FinishTime, std::less<D> > Compare;
  276. Compare fl(finish_time);
  277. std::priority_queue<Vertex, std::vector<Vertex>, Compare > Q(fl);
  278. typename graph_traits<Graph>::vertex_iterator i, j, iend, jend;
  279. boost::tie(i, iend) = vertices(G_T);
  280. boost::tie(j, jend) = vertices(G);
  281. for ( ; i != iend; ++i, ++j) {
  282. put(finish_time, *i, get(finish_time, *j));
  283. Q.push(*i);
  284. }
  285. while ( !Q.empty() ) {
  286. Vertex u = Q.top();
  287. Q.pop();
  288. if (get(color, u) == Color::white()) {
  289. depth_first_visit(G_T, u, vis, color);
  290. ++c_count;
  291. }
  292. }
  293. return c_count;
  294. }
  295. } // namespace boost
  296. #ifdef BOOST_GRAPH_USE_MPI
  297. # include <boost/graph/distributed/strong_components.hpp>
  298. #endif
  299. #endif // BOOST_GRAPH_STRONG_COMPONENTS_HPP