Two-Graphs Common Spanning Trees
Finds all common spanning trees of two undirected graphs using the MRT algorithm of Minty, Read and Tarjan.
Defined in: <boost/graph/two_graphs_common_spanning_trees.hpp>
Example
#include <boost/graph/adjacency_list.hpp>
#include <boost/graph/two_graphs_common_spanning_trees.hpp>
#include <iostream>
#include <vector>
using Graph = boost::adjacency_list<boost::vecS, boost::vecS, boost::undirectedS>;
using Edge = boost::graph_traits<Graph>::edge_descriptor;
int count = 0;
struct CountTrees {
void operator()(const std::vector<bool>& inL) {
++count;
std::cout << " Tree " << count << ": edges";
for (std::size_t i = 0; i < inL.size(); ++i) {
if (inL[i]) { std::cout << " " << i; }
}
std::cout << "\n";
}
};
int main() {
// Two identical triangle graphs (3 vertices, 3 edges)
Graph g1(3), g2(3);
boost::add_edge(0, 1, g1); boost::add_edge(1, 2, g1); boost::add_edge(0, 2, g1);
boost::add_edge(0, 1, g2); boost::add_edge(1, 2, g2); boost::add_edge(0, 2, g2);
std::vector<bool> inL(num_edges(g1), false);
std::cout << "Common spanning trees:\n";
boost::two_graphs_common_spanning_trees(g1, g2, CountTrees{}, inL);
std::cout << "Total: " << count << "\n";
}
Common spanning trees:
Tree 1: edges 0 1
Tree 2: edges 0 2
Tree 3: edges 1 2
Total: 3
(1) Ordered Edges List
template <typename Graph, typename Order,
typename Func, typename Seq>
void two_graphs_common_spanning_trees(
const Graph& iG,
Order iG_map,
const Graph& vG,
Order vG_map,
Func func,
Seq inL);
| Direction | Parameter | Description |
|---|---|---|
IN |
|
The first graph to be analyzed.
The type |
IN |
|
A list of references to edges that defines the preferred order for access to the edge list of |
IN |
|
The second graph to be analyzed.
The type |
IN |
|
A list of references to edges that defines the preferred order for access to the edge list of |
IN |
|
A callback that is invoked by the algorithm for each common spanning tree found.
The type |
IN/OUT |
|
A sequence that marks which edges belong to the common spanning tree.
The type |
(2) Unordered Edges List
template <typename Graph, typename Func,
typename Seq>
void two_graphs_common_spanning_trees(
const Graph& iG,
const Graph& vG,
Func func,
Seq inL);
| Direction | Parameter | Description |
|---|---|---|
IN |
|
The first graph to be analyzed.
The type |
IN |
|
The second graph to be analyzed.
The type |
IN |
|
A callback that is invoked by the algorithm for each common spanning tree found.
The type |
IN/OUT |
|
A sequence that marks which edges belong to the common spanning tree.
The type |
Description
The MRT algorithm, based on an academic article of Minty, Read and Tarjan, is an efficient algorithm for the common spanning tree problem. This kind of algorithm is widely used in electronics, being the basis of the analysis of electrical circuits. Another area of interest may be that of the networking.
The problem of common spanning tree is easily described. Imagine we have two graphs that are represented as lists of edges. A common spanning tree is a set of indices that identifies a spanning tree for both the first and for the second of the two graphs. Despite it is easily accomplished with edge list representation for graphs, it is intuitively difficult to achieve with adjacency list representation. This is due to the fact that it is necessary to represent an edge with an absolute index.
Note that the set of common spanning trees of the two graphs is a subset of the set of spanning trees of the first graph, as well as those of the second graph.
The proposed algorithm receives several arguments and works with callbacks. Overload (1) accepts explicit edge orderings for the two graphs, while overload (2) uses the default edge iteration order.
The file examples/two_graphs_common_spanning_trees.cpp contains an additional example of finding common spanning trees of two undirected graphs.
Notes
[1] The presence of inL may seem senseless. The algorithm can use a vector of placeholders internally generated. However, doing so has more flexibility on the callback function. Moreover, being largely involved in the electronics world, there are cases where some edges have to be forced in every tree (ie you want to search all the trees that have the same root): With this solution, the problem is easily solved. Intuitively from the above, inL must be of a size equal to (V-1), where V is the number of vertices of the graph.