c++ - 使用迭代器时出现编译错误 : "error: ‘...::iterator’ has no member named '...' "

标签 c++ class pointers graph iterator

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

#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’ 的成员 find(it_edge.start),find(it_edge.end),(*(it_edge))->weight)); ^

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

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));
    }

}

最佳答案

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

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

it_edge->start

也可以。

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

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

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

it_edge->start

不起作用。然而,

(*it_edge)->start

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

关于c++ - 使用迭代器时出现编译错误 : "error: ‘...::iterator’ has no member named '...' ",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23113097/

相关文章:

c++ - 在非常量函数中使用 const_iterator

c++ - 为什么 std::reverse 需要双向迭代器?

jquery - 使用 jQuery 更改 CSS 类属性

c# - ...由于其保护级别 c#/asp.net 而无法访问

c++ - 指向记录器类的指针提供给所有其他类?

c++ - 在删除结构之前是否必须删除结构中的指针变量?

c# - C#方法中的静态参数

c++ - 如果您已经有一个 std::shared_ptr 可供使用,那么 std::enable_shared_from_this 有多有用?

c - 将输入文本保存到 main 中的结构

c++ - Visual Studio 2013 强制将 64 DLL 文件转换为 32 位应用程序 - 如何解决此问题?