// Copyright 2004, 2005 The Trustees of Indiana University. // Distributed under the Boost Software License, Version 1.0. // (See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) // Authors: Jeremiah Willcock // Douglas Gregor // Andrew Lumsdaine #ifndef BOOST_GRAPH_ERDOS_RENYI_GENERATOR_HPP #define BOOST_GRAPH_ERDOS_RENYI_GENERATOR_HPP #include #include #include #include #include #include #include #include #include #include #include namespace boost { template class erdos_renyi_iterator : public iterator_facade< erdos_renyi_iterator, std::pair::vertices_size_type, typename graph_traits::vertices_size_type>, std::input_iterator_tag, const std::pair::vertices_size_type, typename graph_traits::vertices_size_type>&> { typedef typename graph_traits::directed_category directed_category; typedef typename graph_traits::vertices_size_type vertices_size_type; typedef typename graph_traits::edges_size_type edges_size_type; BOOST_STATIC_CONSTANT (bool, is_undirected = (is_base_of::value)); public: erdos_renyi_iterator() : gen(), n(0), edges(0), allow_self_loops(false) {} erdos_renyi_iterator(RandomGenerator& gen, vertices_size_type n, double fraction = 0.0, bool allow_self_loops = false) : gen(&gen), n(n), edges(edges_size_type(fraction * n * n)), allow_self_loops(allow_self_loops) { if (is_undirected) edges = edges / 2; next(); } erdos_renyi_iterator(RandomGenerator& gen, vertices_size_type n, edges_size_type m, bool allow_self_loops = false) : gen(&gen), n(n), edges(m), allow_self_loops(allow_self_loops) { next(); } const std::pair& dereference() const { return current; } void increment() { --edges; next(); } bool equal(const erdos_renyi_iterator& other) const { return edges == other.edges; } private: void next() { uniform_int rand_vertex(0, n-1); current.first = rand_vertex(*gen); do { current.second = rand_vertex(*gen); } while (current.first == current.second && !allow_self_loops); } RandomGenerator* gen; vertices_size_type n; edges_size_type edges; bool allow_self_loops; std::pair current; }; template class sorted_erdos_renyi_iterator : public iterator_facade< sorted_erdos_renyi_iterator, std::pair::vertices_size_type, typename graph_traits::vertices_size_type>, std::input_iterator_tag, const std::pair::vertices_size_type, typename graph_traits::vertices_size_type>&> { typedef typename graph_traits::directed_category directed_category; typedef typename graph_traits::vertices_size_type vertices_size_type; typedef typename graph_traits::edges_size_type edges_size_type; BOOST_STATIC_CONSTANT (bool, is_undirected = (is_base_of::value)); public: sorted_erdos_renyi_iterator() : gen(), rand_vertex(0.5), n(0), allow_self_loops(false) , src((std::numeric_limits::max)()), tgt_index(vertices_size_type(-1)), prob(.5) { } // NOTE: The default probability has been changed to be the same as that // used by the geometic distribution. It was previously 0.0, which would // cause an assertion. sorted_erdos_renyi_iterator(RandomGenerator& gen, vertices_size_type n, double prob = 0.5, bool loops = false) : gen(), rand_vertex(1. - prob), n(n), allow_self_loops(loops), src(0) , tgt_index(vertices_size_type(-1)), prob(prob) { this->gen.reset(new uniform_01(&gen)); if (prob == 0.0) {src = (std::numeric_limits::max)(); return;} next(); } const std::pair& dereference() const { return current; } bool equal(const sorted_erdos_renyi_iterator& o) const { return src == o.src && tgt_index == o.tgt_index; } void increment() { next(); } private: void next() { // In order to get the edges from the generator in sorted order, one // effective (but slow) procedure would be to use a // bernoulli_distribution for each legal (src, tgt_index) pair. Because of // the O(|V|^2) cost of that, a geometric distribution is used. The // geometric distribution tells how many times the // bernoulli_distribution would need to be run until it returns true. // Thus, this distribution can be used to step through the edges // which are actually present. BOOST_ASSERT (src != (std::numeric_limits::max)() && src != n); while (src != n) { vertices_size_type increment = rand_vertex(*gen); size_t tgt_index_limit = (is_undirected ? src + 1 : n) + (allow_self_loops ? 0 : -1); if (tgt_index + increment >= tgt_index_limit) { // Overflowed this source; go to the next one and try again. ++src; // This bias is because the geometric distribution always returns // values >=1, and we want to allow 0 as a valid target. tgt_index = vertices_size_type(-1); continue; } else { tgt_index += increment; current.first = src; current.second = tgt_index + (!allow_self_loops && !is_undirected && tgt_index >= src ? 1 : 0); break; } } if (src == n) src = (std::numeric_limits::max)(); } shared_ptr > gen; geometric_distribution rand_vertex; vertices_size_type n; bool allow_self_loops; vertices_size_type src, tgt_index; std::pair current; double prob; }; } // end namespace boost #endif // BOOST_GRAPH_ERDOS_RENYI_GENERATOR_HPP