在C ++中使用STL创建邻接列表的有效方法是什么?

时间:2012-12-31 19:54:13

标签: c++ list graph stl vector

我目前使用的矢量矢量如下:

typedef pair<int, int> vertex;
vector < vector<vertex> > adj_list(n); // n is number of vertices

// Input graph
for (int i = 0; i < edges; i++ )
{
   cin >> source >> target >> weight;
   vertex v(target, weight);
   adj_list[source].push_back(v);
}

列表的向量即。

vector < list<vertex> > adj_list(n);

更好的选择?如果是,为什么?我主要关心的是有效地创建邻接列表,并能够快速读取连接到特定顶点的所有顶点,以实现Dijkstra算法。

3 个答案:

答案 0 :(得分:2)

为此我会使用std :: deque&lt;&gt;,因为你很可能不需要从中间删除元素(这就是为什么有人想要使用std :: list&lt;&gt;)。它应该比std :: vector&lt;&gt;更有效。或std :: list&lt;&gt;。具有连续的内存(向量)和可移动的项目(列表)有它的价格 - 对于列表的向量和指针解除引用/分散的内存的昂贵的调整大小。

另请参阅:http://www.gotw.ca/gotw/054.htm

请注意,如果您的目标是算法竞赛,您可能会对这种基于STL的数据结构可以采用的内存感到惊讶。

答案 1 :(得分:2)

您的需求是快速插入和快速迭代。渐近地,vector<vector<T> >vector<list<T> >之间没有区别:

  • list<T>是一个双向链表,因此每个插页需要O(1)次,每次迭代需要O(1)次。
  • vector<T>是一个数组,实现时每个插入都需要O(1)(摊销)时间[1],迭代每个元素需要O(1)个时间。

操作的常量可能不同,但这是你必须通过分析找到的东西。

但是,空间效率会偏向vector<vector<T> >,因为vector<list<T> >中的每个元素都带有前向和后向指针。所以你可能想要使用vector<vector<T> >,但是你可以避免在常见情况下重新分配(以节省时间),但不要保留太多(以节省空间)。

对于外部向量,您只需在其上调用.reserve(n),其中n是图表中的顶点数。

对于内部向量,它有点难度,它实际上取决于您的数据如何被馈送到此过程。


[1] vector<T>的实现每次重新分配时应将其容量加倍,因此重新分配所需的时间为O(1+2+4+...+n/4+n/2+n) = O(n(1/n+2/n+4/n+...+1/4+1/2+1)) <= O(1+1/2+1/4+...)) = O(2n)。因此分布在n元素上,插入需要O(1)(摊销)时间。

答案 2 :(得分:0)

为图形创建邻接列表的最佳方法是使用“转发列表”(考虑使用C ++中的语言)。如需更多参考,请访问https://www.geeksforgeeks.org/forward-list-c-set-1-introduction-important-functions/

请阅读以下代码以了解“转发列表” 注意: - 在阅读代码之前,请正确理解STL库为“转发列表”提供的实用程序功能。

/* The code has been designed to give output only if ENTER key is pressed, 
before that it'll keep 
recieving inputs from the user. So provide all your inputs in one line 
seperated by whitespaces. */

                      /* This is the implementation of DFS */

#include<iostream>
#include<forward_list>
using namespace std;

class graph{
    private:
        int noOfVertices;
        forward_list<int> *adj;
        void dfsUtil(int v, bool *visited);
    public:
        graph(int v);
        void addEdge(int s, int d);
        void dfs(int startVertex);
};

graph::graph(int v){
    noOfVertices = v;
    adj = new forward_list<int>[noOfVertices];
}

void graph::addEdge(int s, int d){
    adj[s].push_front(d);       
}

void graph::dfs(int startVertex){
    bool *visited = new bool[noOfVertices];

    for(int i = 0 ; i < noOfVertices ; i++){
        visited[i] = false;
        adj[i].reverse();
    }

    dfsUtil(startVertex, visited);
}

void graph::dfsUtil(int v, bool *visited){
    forward_list<int>::iterator i; 
    visited[v] = true;
    cout << v << " ";

    for(i = adj[v].begin() ; i != adj[v].end() ; i++){
        if(!visited[*i])
            dfsUtil(*i, visited);
    } 
}

int main(){
    int v, s, d;
    cin >> v;
    graph g(v);

    while(cin.peek() != '\n')
    {
        cin >> s >> d;
        g.addEdge(s, d);
    }

    g.dfs(2);
}