c++ - 创建OpenCV Mat到2D网格

我正在使用OpenCV Mat构建一个搜索算法,将Mat转换为灰度图像,然后检查像素,用它们的坐标将其签名为walkable或not walkable我使用向量>网格当我试图从网格打印nodeID时,程序突然关闭(例如grid.grid[10][10]->NodeID)。

using namespace std;
int gridZise;
class location{
public:
    int x;
    int y;

};

class Node{
public:
    int gridX;
    int gridY;
    bool walkable;
    location worldPosition;
    int NodeID;


    int gCost;
    int hCost;
    Node *parent;

    Node(bool _walkable, int _gridX, int _gridY)
        {
            walkable = _walkable;
            gridX = _gridX;
            gridY = _gridY;
            NodeID = gridY * gridZise + gridX;
        }
    Node(int _gridX, int _gridY){
        gridX = _gridX;
        gridY = _gridY;
        NodeID = gridY * gridZise + gridX;
    }

    int fCost(){
        return gCost + hCost;
    }

};

class Grid{

public:
    cv::Mat map;
    vector<vector<Node*> > grid;
    int gridx;
    int gridy;


    Grid(cv::Mat _map){
        map = _map;
        gridx = map.cols;
        gridy = map.cols;
        gridZise = map.cols;
    }

    void CreateGrid(){
        // Set up sizes. (HEIGHT x WIDTH)
          grid.resize(gridy);
          for (int i = 0; i < gridy; ++i)
            grid[i].resize(gridx);

          // build up the grid
          for(int i=0; i <gridx;i++){
              for(int j=0; j < gridy;j++){
                  int pixel_val = map.at<int>(i,j);
                  bool _walkable = false;
                  if(pixel_val > 120){//if the value of the pixel is bigger than 120 is walkable
                       _walkable = true;
                  }
                  grid[i][j]->walkable = _walkable;
                  grid[i][j]->gridX = i;
                  grid[i][j]->gridY = j;
              }
          }
    }

    void PrintGrid(){
        for(int i=0; i <gridx;i++){
            for(int j=0; j < gridy;j++){
                cout << grid[i][j]->NodeID <<endl;
            }
        }
    }

    vector<Node> GetNeighbours(Node node)
        {
            vector<Node> neighbours;

            for (int x = -1; x <=1; x++)
            {
                for (int y = -1; y <= 1; y++)
                {
                    if (x == 0 && y == 0)
                        continue;

                    int checkX = node.gridX + x;
                    int checkY = node.gridY + y;

                    if(checkX >=0 && checkX < gridx && checkY >=0 && checkY < gridy)
                    {
                        Node neighbour(checkX,checkY);
                        neighbours.push_back(neighbour);
                    }
                }
            }
            return neighbours;
        }

    Node nodeFromLocation(location _node){
        Node currentNode = *grid[_node.x][_node.y];
        return currentNode;
    }

};


using namespace cv;
int main()
{
    cv::Mat img;
    img = imread("C:\\Users\\abdulla\\Pictures\\maze.jpg");

    if(img.empty()){
        cout<<"image not load "<<endl;
        return -1;
    }
    cvtColor(img,img,CV_BGR2GRAY);
    imshow("image",img);
    waitKey();
    Grid grid(img);

    grid.PrintGrid();

    return 0;
}

谢谢您。


最佳答案:

首先,去掉using namespace std;虽然看起来很方便,但你正在为一些令人不快的惊喜做好准备(参见this问题)。
Grid的构造函数使用Grid::grid的默认构造函数,该构造函数创建一个空向量。
PrintGrid中,您会得到未定义的行为。

grid; // ok, grid is a member of Grid
grid[0]; // Returns a reference. No bounds checking is performed.
grid[0][0]; // undefined behaviour. grid[0] is outside of bounds.

这里有一个方法CreateGrid与此无关,因为您从未调用它但假设你已经打过电话了那么PrintGrid的工作方式如下:
grid; // ok, grid is a member of Grid
grid[0]; // ok
grid[0][0]; // ok, return a pointer to a Node. Which you never initialized.
grid[0][0]->NodeID; // Undefined behaviour. You're trying to access a random memory location.

您真的需要将节点存储为指针吗你也可以使用vector<vector<Node>>这样,就有人(std::vector<...)负责分配和删除节点您仍然可以使用指针指向父级,只要指针用作引用而不是所有者,就可以了。
如果您确实需要存储指针,make use of smart pointers这样,就有人负责删除节点。
最后,类Grid负责保持适当的状态总是所以构造函数应该已经做了CreateGrid所做的事情问题是,您不能从构造函数调用CreateGrid,因为这样您将调用一个对象的方法,该对象的生存期尚未开始(如果我在该方法上出错,有人会纠正我)。
如果你想把它作为一个单独的函数,你可以把CreateGrid设为静态:
class Grid {
    Grid(cv::Mat _map):
           map(_map),
           gridx(_map.cols),
           gridy(_map.cols),
           gridZise(_map.cols),
    {
        GreateGrid(grid, map, gridx, gridy);
    }

//...
    static void CreateGrid(std::vector<std::vector<Node>> & grid, cv::Mat map, int gridx, int grid y) {
    //...
    }
//...
};

译文:来源   文章分类: c++ opencv path-finding a-star

相关文章:

c++ - 推进std :: vector std :: advance VS运算符+的迭代器?

c++ - 使用shared_ptr从方法返回指针是否总能使我免于内存泄漏?

c++ - 分段故障本身就是悬而未决

c++ - C ++派生类和虚拟析构函数

c++ - C ++何时可以扩展`std`命名空间?

c++ - 在向量C ++中删除重复的int数

c++ - 在这种情况下,我可以忽略C4251警告吗?

c++ - 如何对浮点数执行按位运算

c++ - C ++将std :: string复制到char数组,没有空终止

c++ - 布尔在内存中如何表示?