1

我是这个论坛的新手,但我真的可以使用一些帮助。我的任务是用 C++ 编写 Conway 的生命游戏并在 Cygwin 中编译(使用 makefile)。我没有要求任何人为我完成程序或类似的事情,但我完全被困在这一部分......该程序的一个方面是允许用户输入文本文件作为地图初始网格,而不是使用随机生成的网格。这是 .txt 文件格式的示例(数字和“X”纯粹是举例,文件可以是这种格式的任何变体):

5 (rows)
6 (columns)
-----X
X--X--
------
-XX---
------

“X”代表有活细菌的空间,“-”代表死空间。虽然我的程序通过 cygwin 编译得很好,但是当我运行 .exe 时,我得到一个“分段错误(核心转储)”错误。到目前为止,我已经进行了广泛的谷歌搜索,但我发现这个错误通常非常特定于它所关注的程序,所以其他解决方案对我没有多大帮助。我不想用大量代码向你们发送垃圾邮件,所以我现在只包含我的 loadFile 函数。如果您需要更多代码来提供帮助,请告诉我,我会立即发布。到目前为止,这是我的 loadFile 函数中的内容:

void GamePlay::loadFile(int r, int c, char** newBoard){
  int i;
  //int r;
  //int c;
  //char** newBoard;


  int len;
  int k;

  do{
    r = 0;
    c = 0;
    string filePath;
    cout << "Please enter a file path" << endl;
    cin >> filePath;
    DIR* path = opendir(filePath.c_str());
    string fpath = filePath.c_str();
    dirent* openIn = readdir(path);
    string line;
    string ext = ".txt";
    i = 0;

    while(openIn && i != -1){
        if(openIn->d_type == DT_REG){
            string fileName = openIn->d_name;
            string filePath = fpath + fileName;
            int begExt = fileName.find_last_of(".");
            int endExt = fileName.find_last_of("t");
            string extension = fileName.substr(begExt, endExt);

            ifstream in(filePath.c_str());
            k = 0;

            if(in && ext == extension){
                getline(in,line);
                istringstream(line) >> r;
                getline(in,line);
                istringstream(line) >> c;
                newBoard = new char*[r]; //create multi-d array
                for(int a = 0; a < r; ++a){
                    newBoard[a] = new char[c];
                }

                while(in.good() && i != -1){
                    if(k <= r){
                        if(len == c){
                            for(int g = 0; g < r; ++g){
                                getline(in,line);
                                len = line.size();
                                char* arr = new char[len];
                                memcpy(arr,line.c_str(),len);
                                for(int h = 0; h < c; ++h){
                                    newBoard[g][h] = arr[h];
                                }
                            }
                        }
                        /*if(len == c){
                        //newBoard = len*sizeof(char*);
                        for(int a = 0; a < len; ++a){
                        newBoard[a] = r;
                        memcpy(newBoard[a], line, r);
                        }
                        }*/
                        else{
                            cout << "Your column number does not match your grid." << endl;
                            cout << "Please try again with a new file." << endl;
                            i = -1;
                        }
                    }
                    else{
                        cout << "Your row number does not match your grid." << endl;
                        cout << "Please try again with a new file." << endl;
                        i = -1;
                    }
                    k++;
                }
            }
            else{
                cout << "File invalid. Please try again with a new file." << endl;
                i = -1;
            }
            return;
        }
        else{
            cout << "Invalid file path. Please try again with a new file." << endl;
            i = -1;
        }
    }
  }while(i = -1);
}

我也尝试过使用 GDB 进行调试并找到问题所在,但这超出了我的想象。可悲的是,我习惯使用的只是 GUI,例如 Visual Studio 和 Eclipse。任何帮助或建议将不胜感激。谢谢!!!

4

2 回答 2

0

如果您最初的兴趣只是找出代码中的问题所在,那么您不一定需要调试器;我从来没有使用过,通常会设法从我的程序中消除错误。您可以设置标志并使用 cout 语句来确定程序出现问题的位置。当我在高中和大学使用 C++ 时,我广泛地这样做了。

我在下面的代码中输入了一些 cout 语句,以便您知道我在说什么。你还没试过这个,是吗?显然,您运行程序并查看生成了哪些标志输出。如果一个出现而后来一个没有出现,那么您已经缩小了哪一行代码弄乱了您的程序。

    void GamePlay::loadFile(int r, int c, char** newBoard){
      int i;
      //int r;
      //int c;
      //char** newBoard;


      int len;
      int k;

      cout << "flag1";

      do{
        r = 0;
        c = 0;
        string filePath;
        cout << "Please enter a file path" << endl;
        cin >> filePath;

        cout << "flag2";

        DIR* path = opendir(filePath.c_str());
        string fpath = filePath.c_str();
        dirent* openIn = readdir(path);
        string line;
        string ext = ".txt";
        i = 0;

        cout << "flag3";

        while(openIn && i != -1){
            if(openIn->d_type == DT_REG){
                string fileName = openIn->d_name;
                string filePath = fpath + fileName;
                int begExt = fileName.find_last_of(".");
                int endExt = fileName.find_last_of("t");
                string extension = fileName.substr(begExt, endExt);

                cout << "flag4";

                ifstream in(filePath.c_str());
                k = 0;

                if(in && ext == extension){
                    getline(in,line);
                    istringstream(line) >> r;
                    getline(in,line);
                    istringstream(line) >> c;
                    newBoard = new char*[r]; //create multi-d array
                    for(int a = 0; a < r; ++a){
                        newBoard[a] = new char[c];
                    }

                    cout << "flag5";

                    while(in.good() && i != -1){
                        if(k <= r){
                            if(len == c){
                                for(int g = 0; g < r; ++g){
                                    getline(in,line);
                                    len = line.size();
                                    char* arr = new char[len];
                                    memcpy(arr,line.c_str(),len);
                                    for(int h = 0; h < c; ++h){
                                        newBoard[g][h] = arr[h];
                                    }
                                }
                            }
                            /*if(len == c){
                            //newBoard = len*sizeof(char*);
                            for(int a = 0; a < len; ++a){
                            newBoard[a] = r;
                            memcpy(newBoard[a], line, r);
                            }
                            }*/
                            else{
                                cout << "Your column number does not match your grid." << endl;
                                cout << "Please try again with a new file." << endl;
                                i = -1;
                            }
                        }
                        else{
                            cout << "Your row number does not match your grid." << endl;
                            cout << "Please try again with a new file." << endl;
                            i = -1;
                        }
                        k++;
                    }
                }
                else{
                    cout << "File invalid. Please try again with a new file." << endl;
                    i = -1;
                }
                return;
            }
            else{
                cout << "Invalid file path. Please try again with a new file." << endl;
                i = -1;
            }
        }
        cout << "flag6";
      }while(i = -1);
    }
于 2013-09-16T02:03:40.620 回答
0

使用 GDB 非常简单,只需

gdb [path-to-your-executable]
run
[wait for it to crash]
backtrace

这会告诉你坠机发生在哪里。您还可以使用插入断点

break file:line

当它停止时,使用

print [some expression]

要查找有关当前堆栈帧的信息(变量和所有这些),然后“继续”移动到下一个断点。如有必要,您还可以使用远程调试

gdb [path-to-your-executable] [process ID]

如果您的程序向终端屏幕输出大量信息,这很好(因为否则它会与 GDB 的输出混淆)。

于 2013-09-16T01:34:26.343 回答