所以我正在尝试用 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
并评估为分配的值,即 NULL
。
NULL
在用作条件时被视为 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/