c - C 中推箱子游戏加载程序的段错误

标签 c file

所以我正在尝试用 C 语言编写代码,从文件中创建一个字符数组,如下所示:

10 8
##########
######  ##
# $  $  ##
# #  .# ##
#  . #.@ #
##$# *   #
##   #####
##########

但是我遇到了一个段错误,我一直在寻找它可能来自哪里,但没有成功……如果有人能阅读我的代码并给我他/她的反馈,我将不胜感激…… 提前致谢!

//We define a structure representing a map (for the Sokoban game)
    typedef struct map map;
    struct map{
        int width;
        int height;
        int x;
        int y;// x and y are the position of the player
        char* p_char; //p_char is pointing an array which will contain the elements of the map, the characters currently on the file above
    };

    //The function that reads the file and store the characters in an array
    char* ReadMap(const char const* filename)
    {
        FILE* p_file = fopen(filename, "r");
        char* p_array = NULL;
        if (p_file = NULL) {
            exit(EXIT_FAILURE);
        }
        else{   
            size_t size=1;
            while (getc(p_file)!=EOF) {
                size++;
            }
            p_array=(char*)malloc(sizeof(char)*size);
            fseek(p_file,0,SEEK_SET);
            for(size_t i=3; i<size-1; i++) {
                p_array[i]=(char)getc(p_file);//i=3 cause we don't want the 2 first int, representing the size of the array
            }
            p_array[size-1]='\0';
            fclose(p_file);
        }
        return p_array;
    }

    int main(int argc, char *argv[]){
        if (argc != 2) {
            fprintf(stderr, "You must provide a file name!\n");
            exit(EXIT_FAILURE);
        }
        //We define a map structure
        map map_loaded; 
        FILE *p_file1 = NULL;
        p_file1=fopen(argv[1],"r");
        if (p_file1==NULL) {
            fprintf(stderr, "Cannot read file %s!\n", argv[1]);
            exit(EXIT_FAILURE);
        }
        //we're trying to recover width and height, two int at the beginnning of the file
        int width=0;
            map_loaded.width=width;
        int height=0;
            map_loaded.height=height;
        int fscanf_result=0;
            fscanf_result=fscanf(p_file1, "%d %d\n", &width, &height);
        char* p_char=NULL;
        map_loaded.p_char=p_char;
        p_char=ReadMap(argv[1]);
        if (p_char != NULL) {
            printf("%s\n", p_char);
            free(p_char);
        }
    }

最佳答案

段错误的原因是表达式 p_file = NULL 用作条件。

p_file = NULL 是一个赋值 表达式。它将 p_file 设置为 NULL 并评估为分配的值,即 NULLNULL 在用作条件时被视为 false。

另一方面,当条件为比较 p_file == NULL 时,条件为假表示文件打开成功。

在这种情况下,当 p_file 不是 NULL 时,条件变为假。

因此,编写了期望 p_file 不是 NULL 的代码 在 else 子句中。

代码包括将 p_file 传递给 fgetc() 的语句。

结果,NULL 被传递给 fgetc(),这是 Segmentation Fault 的一个可能原因。

请注意,您的代码在解决了这个 Segmentation Fault 之后似乎仍然有问题。

示例中,大小部分为10 8,长度为4个字符。这意味着跳过“3 个字符”没有意义。

另请注意,从 3 开始 i 不会跳过文件内容,只会保留数组的前 3 个元素未初始化,因为某些程序员花花公子指出。

取而代之的是,您可以“跳过第一行”。换句话说,您可以“跳到第一个换行符”。

size_t size=1;
int c;

while (getc(p_file)!=EOF) {
    size++;
}

fseek(p_file,0,SEEK_SET);

while ((c=getc(p_file))!='\n' && c!=EOF) { // drop the first line
    size--;
}

p_array=(char*)malloc(sizeof(char)*size);

if (p_array!=NULL) { // for in case malloc() fails
    for(size_t i=0; i<size-1; i++) {
        p_array[i]=(char)getc(p_file);
    }

    p_array[size-1]='\0';
}

fclose(p_file);

启用 width*i+j 索引的一种方法是通过在 p_array[i]=(char)getc(p_file); 行之后添加此代码来删除换行符>:

if (p_array[i] == '\n') { // if the character is newline, remove it
    i--; // rollback
    size--; // adjust size for the newline character dropped
}

关于c - C 中推箱子游戏加载程序的段错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54944745/

相关文章:

jquery - 使用 jQuery 通过表单数据上传文件时出错

algorithm - 当数据更相似时,是否有二进制数据的哈希函数会产生更接近的哈希值?

c - 为整个地址空间设置保护位

c - 位移位和整数提升?

c# - 为什么我写同样的算法,c#中的代码与c中的代码不同?

c - C函数输入变量

image - 在j2me中如何将图像保存在s40手机内存中?

c++ - 从数据库中搜索时间复杂度为 O(1)\O(log n) 的行

android - 作为 Intent.ACTION_SEND 发送到邮件后删除文件

ajax - 从ajax请求并发写入文件