isomorphism.hpp

来自「CGAL is a collaborative effort of severa」· HPP 代码 · 共 473 行 · 第 1/2 页

HPP
473
字号
            BGL_FORALL_ADJ_T(f[i], v, G2, Graph2)              if (invariant2(v) == invariant1(j) && in_S[v] == false) {                f[j] = v;                in_S[v] = true;                num_edges_on_k = 1;                BOOST_USING_STD_MAX();                int next_k = max BOOST_PREVENT_MACRO_SUBSTITUTION(dfs_num_k, max BOOST_PREVENT_MACRO_SUBSTITUTION(dfs_num[i], dfs_num[j]));                if (match(next(iter), next_k))                  return true;                in_S[v] = false;              }                                          }          else {            if (contains(adjacent_vertices(f[i], G2), f[j])) {              ++num_edges_on_k;              if (match(next(iter), dfs_num_k))                return true;            }                          }        } else           return true;        return false;      }        };        template <typename Graph, typename InDegreeMap>    void compute_in_degree(const Graph& g, InDegreeMap in_degree_map)    {      BGL_FORALL_VERTICES_T(v, g, Graph)        put(in_degree_map, v, 0);      BGL_FORALL_VERTICES_T(u, g, Graph)        BGL_FORALL_ADJ_T(u, v, g, Graph)        put(in_degree_map, v, get(in_degree_map, v) + 1);    }  } // namespace detail  template <typename InDegreeMap, typename Graph>  class degree_vertex_invariant  {    typedef typename graph_traits<Graph>::vertex_descriptor vertex_t;    typedef typename graph_traits<Graph>::degree_size_type size_type;  public:    typedef vertex_t argument_type;    typedef size_type result_type;    degree_vertex_invariant(const InDegreeMap& in_degree_map, const Graph& g)      : m_in_degree_map(in_degree_map), m_g(g) { }    size_type operator()(vertex_t v) const {      return (num_vertices(m_g) + 1) * out_degree(v, m_g)        + get(m_in_degree_map, v);    }    // The largest possible vertex invariant number    size_type max BOOST_PREVENT_MACRO_SUBSTITUTION () const {       return num_vertices(m_g) * num_vertices(m_g) + num_vertices(m_g);    }  private:    InDegreeMap m_in_degree_map;    const Graph& m_g;  };  template <typename Graph1, typename Graph2, typename IsoMapping,     typename Invariant1, typename Invariant2,    typename IndexMap1, typename IndexMap2>  bool isomorphism(const Graph1& G1, const Graph2& G2, IsoMapping f,                    Invariant1 invariant1, Invariant2 invariant2,                    std::size_t max_invariant,                   IndexMap1 index_map1, IndexMap2 index_map2)  {    // Graph requirements    function_requires< VertexListGraphConcept<Graph1> >();    function_requires< EdgeListGraphConcept<Graph1> >();    function_requires< VertexListGraphConcept<Graph2> >();    function_requires< BidirectionalGraphConcept<Graph2> >();        typedef typename graph_traits<Graph1>::vertex_descriptor vertex1_t;    typedef typename graph_traits<Graph2>::vertex_descriptor vertex2_t;    typedef typename graph_traits<Graph1>::vertices_size_type size_type;        // Vertex invariant requirement    function_requires< AdaptableUnaryFunctionConcept<Invariant1,      size_type, vertex1_t> >();    function_requires< AdaptableUnaryFunctionConcept<Invariant2,      size_type, vertex2_t> >();        // Property map requirements    function_requires< ReadWritePropertyMapConcept<IsoMapping, vertex1_t> >();    typedef typename property_traits<IsoMapping>::value_type IsoMappingValue;    BOOST_STATIC_ASSERT((is_same<IsoMappingValue, vertex2_t>::value));        function_requires< ReadablePropertyMapConcept<IndexMap1, vertex1_t> >();    typedef typename property_traits<IndexMap1>::value_type IndexMap1Value;    BOOST_STATIC_ASSERT((is_convertible<IndexMap1Value, size_type>::value));        function_requires< ReadablePropertyMapConcept<IndexMap2, vertex2_t> >();    typedef typename property_traits<IndexMap2>::value_type IndexMap2Value;    BOOST_STATIC_ASSERT((is_convertible<IndexMap2Value, size_type>::value));        if (num_vertices(G1) != num_vertices(G2))      return false;    if (num_vertices(G1) == 0 && num_vertices(G2) == 0)      return true;        detail::isomorphism_algo<Graph1, Graph2, IsoMapping, Invariant1,      Invariant2, IndexMap1, IndexMap2>       algo(G1, G2, f, invariant1, invariant2, max_invariant,            index_map1, index_map2);    return algo.test_isomorphism();  }  namespace detail {      template <typename Graph1, typename Graph2,       typename IsoMapping,       typename IndexMap1, typename IndexMap2,      typename P, typename T, typename R>    bool isomorphism_impl(const Graph1& G1, const Graph2& G2,                           IsoMapping f, IndexMap1 index_map1, IndexMap2 index_map2,                          const bgl_named_params<P,T,R>& params)    {      std::vector<std::size_t> in_degree1_vec(num_vertices(G1));      typedef safe_iterator_property_map<std::vector<std::size_t>::iterator,                                         IndexMap1#ifdef BOOST_NO_STD_ITERATOR_TRAITS                                         , std::size_t, std::size_t&#endif /* BOOST_NO_STD_ITERATOR_TRAITS */                                         > InDeg1;      InDeg1 in_degree1(in_degree1_vec.begin(), in_degree1_vec.size(), index_map1);      compute_in_degree(G1, in_degree1);      std::vector<std::size_t> in_degree2_vec(num_vertices(G2));      typedef safe_iterator_property_map<std::vector<std::size_t>::iterator,                                          IndexMap2#ifdef BOOST_NO_STD_ITERATOR_TRAITS                                         , std::size_t, std::size_t&#endif /* BOOST_NO_STD_ITERATOR_TRAITS */                                         > InDeg2;      InDeg2 in_degree2(in_degree2_vec.begin(), in_degree2_vec.size(), index_map2);      compute_in_degree(G2, in_degree2);      degree_vertex_invariant<InDeg1, Graph1> invariant1(in_degree1, G1);      degree_vertex_invariant<InDeg2, Graph2> invariant2(in_degree2, G2);      return isomorphism(G1, G2, f,                         choose_param(get_param(params, vertex_invariant1_t()), invariant1),                         choose_param(get_param(params, vertex_invariant2_t()), invariant2),                         choose_param(get_param(params, vertex_max_invariant_t()), (invariant2.max)()),                         index_map1, index_map2                         );      }       } // namespace detail  // Named parameter interface  template <typename Graph1, typename Graph2, class P, class T, class R>  bool isomorphism(const Graph1& g1,                   const Graph2& g2,                   const bgl_named_params<P,T,R>& params)  {    typedef typename graph_traits<Graph2>::vertex_descriptor vertex2_t;    typename std::vector<vertex2_t>::size_type n = num_vertices(g1);    std::vector<vertex2_t> f(n);    return detail::isomorphism_impl      (g1, g2,        choose_param(get_param(params, vertex_isomorphism_t()),                    make_safe_iterator_property_map(f.begin(), f.size(),                                                    choose_const_pmap(get_param(params, vertex_index1),                                                                      g1, vertex_index), vertex2_t())),       choose_const_pmap(get_param(params, vertex_index1), g1, vertex_index),       choose_const_pmap(get_param(params, vertex_index2), g2, vertex_index),       params       );  }  // All defaults interface  template <typename Graph1, typename Graph2>  bool isomorphism(const Graph1& g1, const Graph2& g2)  {    return isomorphism(g1, g2,                       bgl_named_params<int, buffer_param_t>(0));// bogus named param  }  // Verify that the given mapping iso_map from the vertices of g1 to the  // vertices of g2 describes an isomorphism.  // Note: this could be made much faster by specializing based on the graph  // concepts modeled, but since we're verifying an O(n^(lg n)) algorithm,  // O(n^4) won't hurt us.  template<typename Graph1, typename Graph2, typename IsoMap>  inline bool verify_isomorphism(const Graph1& g1, const Graph2& g2, IsoMap iso_map)  {#if 0    // problematic for filtered_graph!    if (num_vertices(g1) != num_vertices(g2) || num_edges(g1) != num_edges(g2))      return false;#endif      for (typename graph_traits<Graph1>::edge_iterator e1 = edges(g1).first;         e1 != edges(g1).second; ++e1) {      bool found_edge = false;      for (typename graph_traits<Graph2>::edge_iterator e2 = edges(g2).first;           e2 != edges(g2).second && !found_edge; ++e2) {        if (source(*e2, g2) == get(iso_map, source(*e1, g1)) &&            target(*e2, g2) == get(iso_map, target(*e1, g1))) {          found_edge = true;        }      }          if (!found_edge)        return false;    }      return true;  }} // namespace boost#ifdef BOOST_ISO_INCLUDED_ITER_MACROS#undef BOOST_ISO_INCLUDED_ITER_MACROS#include <boost/graph/iteration_macros_undef.hpp>#endif#endif // BOOST_GRAPH_ISOMORPHISM_HPP

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?