C语言中Sokoban游戏加载程序的分段错误

C语言中Sokoban游戏加载程序的分段错误,c,file,C,File,因此,我试图用C编写一个代码,从一个文件中创建一个字符数组,如下所示: 10 8 ########## ###### ## # $ $ ## # # .# ## # . #.@ # ##$# * # ## ##### ########## 但我被一个分割错误卡住了,我一直在寻找它可能来自哪里,但没有成功。。。如果有人能阅读我的代码并给我反馈,我将不胜感激。。。 提前谢谢 //We define a structure representing a map (for the S

因此,我试图用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);
        }
    }
//我们定义了一个表示地图的结构(用于Sokoban游戏)
typedef结构图;
结构图{
整数宽度;
内部高度;
int x;
int y;//x和y是玩家的位置
char*p_char;//p_char指向一个数组,该数组将包含映射的元素,即上面文件中当前的字符
};
//读取文件并将字符存储在数组中的函数
char*ReadMap(const char const*filename)
{
FILE*p_FILE=fopen(文件名,“r”);
char*p_数组=NULL;
如果(p_文件=NULL){
退出(退出失败);
}
否则{
尺寸=1;
while(getc(p_文件)!=EOF){
大小++;
}
p_数组=(char*)malloc(sizeof(char)*size);
fseek(p_文件,0,SEEK_集);

对于(size_t i=3;i分段错误的原因是用作条件的表达式
p_file=NULL

p_file=NULL
是一个赋值表达式。它将
p_file
设置为
NULL
,并计算为赋值,即
NULL
NULL
用作条件时被视为false

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

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

因此,将编写期望
p_文件
不是
NULL
的代码 在
else
子句中

代码包括将
p\u文件
传递到
fgetc()
的语句

因此,
NULL
被传递给
fgetc()
,这是分段错误的一个可能原因

请注意,在消除此分段错误后,代码似乎仍然是错误的

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

还请注意,从3开始
i
不会跳过文件内容,只会将数组的前3个元素保留为未初始化的元素,因为这是一些程序员的工作点

相反,你可以“跳过第一行”,换句话说,你可以“跳过直到第一个换行符”


如果(p_file=NULL)将NULL赋值给p_fill,那么测试应该使用==not=您将分配的数据的前三个元素保留为未初始化。并且您仍然将第一行的维度读取到数据中。第一行,即包含维度的行,将是五个字符,而不是三个字符(在您显示的示例中)。您需要重新考虑如何读取文件内容。在从文件中读取数据之前,您设置了
map\u loaded.width=width;
,同上
height
。为什么不检查fscanf\u结果?1.将
NULL
分配给文件指针,而不是进行比较。2.
NULL
被视为false,因此程序未运行假设文件读取成功,并尝试使用文件指针,该指针被意外分配到了
NULL
。3.
NULL
被传递到文件,将函数视为文件指针。4.哎哟!感谢所有的答案!我已经更正了分配错误,这是一个简单的注意错误…但正如你所说,我仍然无法处理“绕过”我文件的第一个int,如何从第一个开始循环?@candidethoveso我将它添加到我的答案中。非常感谢,我不知道while的诀窍(c=getc(p#file))!='\n'),将在下一次记下它!不过,最后一个问题,我想通过访问索引宽度为*i+j的数组中的元素来访问第i行和第j列的元素,但似乎无法使用函数返回的数组…我一直在试图弄清楚为什么返回的数组是二维的,但我不明白。。.有什么想法吗?@candidethoveso可以通过删除额外的字符来实现索引。示例代码添加到我的答案中。非常感谢,效果非常好!也不知道if(p_数组[i]='\n')!
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);
if (p_array[i] == '\n') { // if the character is newline, remove it
    i--; // rollback
    size--; // adjust size for the newline character dropped
}