简体   繁体   English

如何在Boost Graph中访问边缘信息?

[英]How to access edge information in Boost Graph?

The main question: 主要问题:

I am able to create a graph implementation with information structs assigned to the vertices and edges: 我能够使用分配给顶点和边的信息结构来创建图实现:

struct vertex_info {std::string name;};
struct edge_info {std::string name;};

typedef boost::adjacency_list<
  boost::vecS,
  boost::vecS,
  boost::undirectedS,
  vertex_info,
  edge_info> UndirectedGraph;

And for an instance of UndirectedGraph , g , I can easily iterate over the vertices, and access their information: 对于UndirectedGraph的实例g ,我可以轻松地遍历顶点并访问其信息:

for(size_t i=0; i<boost::num_vertices(g); i++){
  std::cout << g[i].name << std::endl;
}

but I am unable to figure out how to do the same for the edges. 但是我无法弄清楚边缘的处理方法。 I have come across some iterators to loop over all the edges, but I cannot access these edges as some kind of object or something with properties. 我遇到了一些迭代器来遍历所有边缘,但是我无法将这些边缘作为某种对象或具有属性的对象来访问。 How can I access the edge information of g ? 如何访问g的边缘信息?

A minimal working demonstration: 最小的工作演示:

#include <iostream>
#include <utility>
#include <vector>
#include <string>

#include "boost/graph/graph_traits.hpp"
#include "boost/graph/adjacency_list.hpp"

int main(int argc, char *argv[])
{

  //Add vertex information struct
  struct vertex_info {
    std::string name;
  };

  //Add edge information struct
  struct edge_info {
    std::string name;
  };

  //Typedef my graph implimentation
  typedef boost::adjacency_list<boost::vecS, boost::vecS, boost::undirectedS, vertex_info, edge_info> UndirectedGraph;

  //Our set of edges, and count N: (0-7) and 8
  enum {C, D, G, I, S, J, L, H, N};
  const char *name = "CDGISJLH";

  //Create a vector of edges
  typedef std::pair<int, int> Edge;
  std::vector<Edge> edgeVec;
  edgeVec.push_back(Edge(C,D));
  edgeVec.push_back(Edge(D,G));
  edgeVec.push_back(Edge(I,G));
  edgeVec.push_back(Edge(G,L));
  edgeVec.push_back(Edge(H,G));
  edgeVec.push_back(Edge(I,S));
  edgeVec.push_back(Edge(S,J));
  edgeVec.push_back(Edge(L,J));
  edgeVec.push_back(Edge(H,J));

  //Now we can initialize our graph using iterators from our above vector
  UndirectedGraph g(edgeVec.begin(), edgeVec.end(), N);

  std::cout << num_edges(g) << "\n";  //Outputs: 9

  //loop over vertices, access "name" property
  for(size_t i=0; i<boost::num_vertices(g); i++){
    //And add information to the edges
    g[i].name = "foo";
  }

  //We can access the first vertice and print the property
  std::cout << g[0].name << std::endl; //Outputs: foo

  //Edge iterator for or graph
  typedef boost::graph_traits<UndirectedGraph>::edge_iterator edge_iterator;

  //Iterate through all the edges
  std::pair<edge_iterator, edge_iterator> ei = boost::edges(g);
  for(edge_iterator edge_iter = ei.first; edge_iter != ei.second; ++edge_iter) {
   //How can I access the edge property???

  }
}

May not be exactly what you are looking for but does achieve what you are after 可能不完全是您想要的,但确实可以实现您想要的

#include <iostream>
#include <utility>
#include <vector>
#include <string>

#include <boost/graph/graph_traits.hpp>
#include <boost/graph/adjacency_list.hpp>

namespace boost {
enum edge_myname_t { edge_myname };
BOOST_INSTALL_PROPERTY(boost::edge, myname);
}

int main(int argc, char* argv[]) {
  // Add vertex information struct
  struct vertex_info {
    std::string name;
  };

  // Add edge information struct
  //struct edge_info {
    //std::string name;
  //};

  using EdgeName = boost::property<boost::edge_myname_t, std::string>;

  // Typedef my graph implimentation
  using UndirectedGraph =
      boost::adjacency_list<boost::vecS, boost::vecS, boost::undirectedS,
                            vertex_info, EdgeName>;

  // Our set of edges, and count N: (0-7) and 8
  enum { C, D, G, I, S, J, L, H, N };
  const char* name = "CDGISJLH";

  // Create a vector of edges
  //using Edge = std::pair<int, int>;
  //std::vector<Edge> edgeVec;
  //edgeVec.push_back(Edge(C, D));
  //edgeVec.push_back(Edge(D, G));
  //edgeVec.push_back(Edge(I, G));
  //edgeVec.push_back(Edge(G, L));
  //edgeVec.push_back(Edge(H, G));
  //edgeVec.push_back(Edge(I, S));
  //edgeVec.push_back(Edge(S, J));
  //edgeVec.push_back(Edge(L, J));
  //edgeVec.push_back(Edge(H, J));

  // Now we can initialize our graph using iterators from our above vector
  UndirectedGraph g(N);
  //UndirectedGraph g(edgeVec.begin(), edgeVec.end(), N);

  boost::add_edge(C, D, EdgeName("#1"), g);
  boost::add_edge(D, G, EdgeName("#2"), g);
  boost::add_edge(I, G, EdgeName("#3"), g);
  boost::add_edge(G, L, EdgeName("#4"), g);
  boost::add_edge(H, G, EdgeName("#5"), g);
  boost::add_edge(I, S, EdgeName("#6"), g);
  boost::add_edge(S, J, EdgeName("#7"), g);
  boost::add_edge(L, J, EdgeName("#8"), g);
  boost::add_edge(H, J, EdgeName("#9"), g);

  boost::property_map<UndirectedGraph, boost::edge_myname_t>::type get_name =
      boost::get(boost::edge_myname, g);

  std::cout << num_edges(g) << "\n";  // Outputs: 9

  // loop over vertices, access "name" property
  for (size_t i = 0; i < boost::num_vertices(g); i++) {
    // And add information to the edges
    g[i].name = "foo";
  }

  // We can access the first vertice and print the property
  std::cout << g[0].name << std::endl;  // Outputs: foo

  // Edge iterator for or graph
  using EdgeIterator = boost::graph_traits<UndirectedGraph>::edge_iterator;

  // Iterate through all the edges
  std::pair<EdgeIterator, EdgeIterator> ei = boost::edges(g);
  for (EdgeIterator edge_iter = ei.first; edge_iter != ei.second; ++edge_iter) {
    // How can I access the edge property???
    std::cout << get_name[*edge_iter] << "\n";
  }
}

I just slightly modified some of the code for my own readability issues. 我只是针对自己的可读性问题对一些代码进行了一些修改。

For reference, check this out. 作为参考,检查出。

I have figure out the problem by walking through this example: https://www.boost.org/doc/libs/1_71_0/libs/graph/doc/bundles.html 我通过遍历此示例来找出问题所在: https : //www.boost.org/doc/libs/1_71_0/libs/graph/doc/bundles.html

The fix: 解决方法:

Although I still don't exactly understand how it all works. 尽管我仍然不太清楚这是如何工作的。 It seems like you have to use edge_iter as some kind of index into g : 似乎您必须使用edge_iter作为g某种索引:

//Edge iterator for or graph
typedef boost::graph_traits<MRFGraph>::edge_iterator edge_iterator;

//Iterate through all the edges
std::pair<edge_iterator, edge_iterator> ei = boost::edges(g);
for(edge_iterator edge_iter = ei.first; edge_iter != ei.second; ++edge_iter) {
  g[*edge_iter].name = "bar";
  std::cout << *edge_iter << ": " << g[*edge_iter].name << std::endl;
}

Output: 输出:

If I add this to the minimal working demonstration, it produces the following output: 如果我将此添加到最小的工作演示中,它将产生以下输出:

9
foo
(0,1): bar
(1,2): bar
(3,2): bar
(2,6): bar
(7,2): bar
(3,4): bar
(4,5): bar
(6,5): bar
(7,5): bar

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM