【问题标题】:Compiler error using iterator: " error: ‘... ::iterator’ has no member named '...' "使用迭代器的编译器错误:“错误:'... ::iterator' 没有名为 '...' 的成员”
【发布时间】:2014-04-16 14:58:25
【问题描述】:

下面是代码。更多相关位位于 Class Edge 和 Vertex 下,错误抛出复制构造函数位于初始块下方。 我确定这与我调用 end 和 start 的方式有关,看看它们在 Edge() 类中的 Vertex* 类型如何,并且我使用 std:: 类型的迭代器调用它们矢量::边*>。这是基于指针的东西。 是的,这是针对课堂的,但这是一个包含更多内容的课堂项目,这只是我遇到的一个错误。提前谢谢,我可能只是因为睡眠不足或其他原因而没有看到明显的东西。

#ifndef GRAPH_H
#define GRAPH_H

#include <vector>
#include <limits>
#include <string>
#include <iostream>
#include <fstream>

using namespace std;

template <typename Object, typename Weight>
class graph {
public:

class Vertex;

class Edge {
  public:
    Edge(Vertex* v, Vertex* w, Weight setweight) {
      start = v;
      end = w;
      v->edge.push_back(this);
      w->inedge.push_back(this);
      weight = setweight;
      explored = false;
    }
    Edge() {
      explored = false;
    }
    Weight weight;
    Vertex* start;
    Vertex* end;
    bool explored;
};

class Vertex {
  public:
    Vertex(Object setelement) {
      level = 0;
      connectedcomponent = 0;
      element = setelement;
      back = NULL;
      explored = false;
    }
    Vertex() {
      level = 0;
      connectedcomponent = 0;
      back = NULL;
      explored = false;
    }
    Object element;
    vector<Edge*> edge;
    vector<Edge*> inedge;
    double value;
    size_t starttime, finishtime;
    size_t level;
    size_t connectedcomponent;
    float rank;
    Vertex* back;
    int color;
    bool explored;
};

private:

vector<Edge*> edge;

vector<Vertex*> vertex;

size_t counter;

public:
graph();

graph(graph& G);

~graph();

void reset();
void resetBack();
void resetValues();
void resetLevels();
void resetExplored();
void resetConnectedComponents();
vector<Vertex*> incidentVertices(Vertex* v);
vector<Edge*> incidentEdges(Vertex* v);
vector<Edge*> outgoingEdges(Vertex* v);
vector<Vertex*> adjacentVertices(Vertex* v);
size_t indegree(Vertex* v);
size_t outdegree(Vertex* v);
size_t degree(Vertex* v);
Vertex* startVertex(Edge* e);
Vertex* endVertex(Edge* e);

bool isAdjacent(Vertex* v, Vertex* w);

Vertex* insertVertex(Object o);
void insertEdge(Vertex* v, Vertex* w, Weight t);
void insertUndirectedEdge(Vertex* v, Vertex* w, Weight t);
void removeVertex(Vertex* v);
void removeEdge(Edge* e);

size_t numVertices();
size_t numEdges();
vector<Vertex*> vertices();
vector<Edge*> edges();

void print();
void read_file(std::string filename);
};

template <typename Object, typename Weight>
graph<Object, Weight>::graph() {//Default Constructor
    Edge();
    Vertex();
    counter = 0;
}

复制构造函数,它会生成错误(我知道这是基于迭代器的)

Graph.h:149:22: 错误:‘std::vector::Edge*, std::allocator::Edge*> >::iterator’没有名为‘start’的成员 查找(it_edge.start),查找(it_edge.end),(*(it_edge))->重量)); ^

Graph.h:149:40: 错误:‘std::vector::Edge*, std::allocator::Edge*> >::iterator’没有名为‘end’的成员 查找(it_edge.start),查找(it_edge.end),(*(it_edge))->重量)); ^

template <typename Object, typename Weight>
graph<Object, Weight>::graph(graph<Object, Weight>& G) {//Copy Constructor

    typename std::vector<graph<Object, Weight>::Vertex*>::iterator it_vert;

    for(it_vert = vertex.begin(); it_vert != vertex.end(); ++it_vert){
      vertex.push_back(new Vertex((*(it_vert))->element));
    }

    typename std::vector<graph<Object, Weight>::Edge*>::iterator it_edge;

    for(it_edge = edge.begin(); it_edge != edge.end(); ++it_edge){
      edge.push_back(new Edge(
      find(it_edge.start),find(it_edge.end),(*(it_edge))->weight));
    }

}

【问题讨论】:

  • 我试过了,但得到了 Graph.h:149:23: error: request for member 'start' in '* it_edge.__gnu_cxx::__normal_iterator<_iterator _container>::operator->::Edge**, std::vector::Edge*, std::allocator::Edge*> > ()',它是指针类型'graph::Edge*' 我已经没有关于此时应该如何调用 start 或 end 的想法
  • 也许你可以使用boost图形库?它有很多处理图表的工具。
  • @user3210680 嗯,似乎有一些来自 ref 返回的迭代器的干扰。 (*it_edge)-&gt;start 有效。我会尝试找出为什么我原来的建议不起作用。

标签: c++ class pointers graph iterator


【解决方案1】:
find(it_edge.start)

这会尝试访问对象it_edge 的数据成员startit_edge 是一个迭代器:

typename std::vector<graph<Object, Weight>::Edge*>::iterator it_edge;

您可能希望访问此迭代器(间接)指向的Edge 对象的数据成员start

如果您通过* 对迭代器应用间接寻址,您将获得(引用)迭代器指向的对象:

*it_edge // returns an Edge*

您需要第二次间接获取此指针指向的Edge 对象:

**it_edge // returns an Edge object (lvalue)

然后,就可以访问这个对象的数据成员start了:

(**it_edge).start

通常,-&gt; 会递归应用,直到操作数是指针。所以我一开始以为

it_edge->start

也可以。

然而,标准库迭代器需要将a-&gt;m 实现为(*a).m [input.iterators]/Table 107。因此,迭代器类返回 一个指向其元素的指针重载operator-&gt;:

template<class T>
class iterator
{
private:
    T* ptr;
public:
    T* operator->() { return ptr; }
};

在 OP 中,我们有 T == Edge*,所以 T* == Edge**,这就是原因

it_edge->start

不起作用。然而,

(*it_edge)->start

工作正常,因为它还应用了两个必要的间接。

【讨论】:

  • 抱歉,我超出了 wifi 范围。我之前也试过一次,它给了我这个详细的错误 Graph.h:149:26: error: no matching function for call to 'find(graph::Vertex*&)' find((*it_edge )->start),find((*it_edge)->end),(*(it_edge))->weight)); ^ Graph.h:149:26:注意:候选者是:在 /usr/include/c++/4.8/bits/locale_facets.h:48:0 包含的文件中,来自 /usr/include/c++/4.8/bits/basic_ios .h:37, ...等等......粘贴所有这些运行超过限制的大约 3.2k 个字符。错误大多与此相同
  • 是的,我在尝试编译您的程序时也遇到了这个错误。我以为你在这里只展示了一部分。你想调用哪个函数find
  • 我正在调用 std::find()。这似乎是问题所在。如果我确实正确调用了指针,它试图在 Edge* 向量中找到一个 Vertex*。我最终更改了解决问题的代码实现,但不幸的是,我没有得到我想要的深层副本。不管怎样,手头的问题已经解决了,非常感谢。
  • 调用std::find 需要两个参数(例如find((**it_edge).start, (**it_edge).end)),并且需要包含&lt;algorithm&gt; 标头。至于深层副本:如果您不在 graph 类中使用拥有原始指针(例如 vector&lt;Edge*&gt; edge; vector&lt;Vertex*&gt; vertex;),那会容易得多。
  • 哦,是的,我确实尝试过,但通过边向量找到顶点并不开心。我将从这里重新考虑我的深层副本
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2012-02-03
  • 1970-01-01
  • 2014-03-03
  • 1970-01-01
  • 2012-09-22
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多