Xcode - EXC_BAD_ACCESS 上的 C 编程
C Programming on Xcode - EXC_BAD_ACCESS
我目前正在编写一个 C 程序来创建、查看和更新文件中的员工记录。我的程序没有严重错误,在我输入员工 ID 之前一切都顺利编译,Xcode 会给出 EXC_BAD_ACCESS 错误。
我回顾了我的代码,这发生在应该将数据写入文件的部分(请参阅评论),但我仍然不确定这里出了什么问题。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define size 300
struct emp
{
int id;
char *name;
}*emp1, *emp3;
void display();
void create();
void update();
FILE *fp, *fp1;
int count = 0;
void main(int argc, char **argv)
{
int i, n, ch;
printf("1] Create a Record\n");
printf("2] Display Records\n");
printf("3] Update Records\n");
printf("4] Exit");
while (1)
{
printf("\nEnter your choice : ");
scanf("%d", &ch);
switch (ch)
{
case 1:
fp = fopen(argv[1], "a");
create();
break;
case 2:
fp1 = fopen(argv[1],"rb");
display();
break;
case 3:
fp1 = fopen(argv[1], "r+");
update();
break;
case 4:
exit(0);
}
}
}
/* To create an employee record */
void create()
{
int i;
char *p;
emp1 = (struct emp *)malloc(sizeof(struct emp));
emp1->name = (char *)malloc((size)*(sizeof(char)));
printf("Enter name of employee : ");
scanf(" %[^\n]s", emp1->name);
printf("Enter emp id : ");
scanf(" %d", &emp1->id);
fwrite(&emp1->id, sizeof(emp1->id), 1, fp); //The error occurs here
fwrite(emp1->name, size, 1, fp);
count++; // count to number of entries of records
fclose(fp);
}
/* Display the records in the file */
void display()
{
emp3=(struct emp *)malloc(1*sizeof(struct emp));
emp3->name=(char *)malloc(size*sizeof(char));
int i = 1;
if (fp1 == NULL)
printf("\nFile not opened for reading");
while (i <= count)
{
fread(&emp3->id, sizeof(emp3->id), 1, fp1);
fread(emp3->name, size, 1, fp1);
printf("\n%d %s",emp3->id,emp3->name);
i++;
}
fclose(fp1);
free(emp3->name);
free(emp3);
}
void update()
{
int id, flag = 0, i = 1;
char s[size];
if (fp1 == NULL)
{
printf("File cant be opened");
return;
}
printf("Enter employee id to update : ");
scanf("%d", &id);
emp3 = (struct emp *)malloc(1*sizeof(struct emp));
emp3->name=(char *)malloc(size*sizeof(char));
while(i<=count)
{
fread(&emp3->id, sizeof(emp3->id), 1, fp1);
fread(emp3->name,size,1,fp1);
if (id == emp3->id)
{
printf("Enter new name of emplyee to update : ");
scanf(" %[^\n]s", s);
fseek(fp1, -204L, SEEK_CUR);
fwrite(&emp3->id, sizeof(emp3->id), 1, fp1);
fwrite(s, size, 1, fp1);
flag = 1;
break;
}
i++;
}
if (flag != 1)
{
printf("No employee record found");
flag = 0;
}
fclose(fp1);
free(emp3->name); /* to free allocated memory */
free(emp3);
}
您在评论中提到 fp
为 NULL。这意味着您实际上没有要写入的打开文件;较早的 fopen
调用 returned NULL 以指示错误。崩溃的 直接 原因是 fwrite
通过尝试访问它应该指向的 FILE
结构的字段来取消引用空指针; 间接 原因是您在使用前没有检查 fopen
中的 return 值。
在您的每个 fopen
调用之后,添加如下内容:
if (fp == NULL)
{
perror("Failed to open file");
exit(1);
}
当 fopen
失败时,它 return 为 NULL 并设置全局变量 errno
以指示出了什么问题。 perror
函数打印与 errno
值相对应的描述性消息。
由于您的程序希望在命令行中获得一个文件名(即在 argv[1]
中,请确保 Xcode 实际上给了它一个文件名。从 IDE 启动程序默认情况下通常在没有任何命令行参数的情况下运行它。当 argc
小于 2 时访问 argv[1]
将是另一个错误,并且可能会显示为不是有效文件名的垃圾字符串,导致fopen
失败。
我目前正在编写一个 C 程序来创建、查看和更新文件中的员工记录。我的程序没有严重错误,在我输入员工 ID 之前一切都顺利编译,Xcode 会给出 EXC_BAD_ACCESS 错误。
我回顾了我的代码,这发生在应该将数据写入文件的部分(请参阅评论),但我仍然不确定这里出了什么问题。
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define size 300
struct emp
{
int id;
char *name;
}*emp1, *emp3;
void display();
void create();
void update();
FILE *fp, *fp1;
int count = 0;
void main(int argc, char **argv)
{
int i, n, ch;
printf("1] Create a Record\n");
printf("2] Display Records\n");
printf("3] Update Records\n");
printf("4] Exit");
while (1)
{
printf("\nEnter your choice : ");
scanf("%d", &ch);
switch (ch)
{
case 1:
fp = fopen(argv[1], "a");
create();
break;
case 2:
fp1 = fopen(argv[1],"rb");
display();
break;
case 3:
fp1 = fopen(argv[1], "r+");
update();
break;
case 4:
exit(0);
}
}
}
/* To create an employee record */
void create()
{
int i;
char *p;
emp1 = (struct emp *)malloc(sizeof(struct emp));
emp1->name = (char *)malloc((size)*(sizeof(char)));
printf("Enter name of employee : ");
scanf(" %[^\n]s", emp1->name);
printf("Enter emp id : ");
scanf(" %d", &emp1->id);
fwrite(&emp1->id, sizeof(emp1->id), 1, fp); //The error occurs here
fwrite(emp1->name, size, 1, fp);
count++; // count to number of entries of records
fclose(fp);
}
/* Display the records in the file */
void display()
{
emp3=(struct emp *)malloc(1*sizeof(struct emp));
emp3->name=(char *)malloc(size*sizeof(char));
int i = 1;
if (fp1 == NULL)
printf("\nFile not opened for reading");
while (i <= count)
{
fread(&emp3->id, sizeof(emp3->id), 1, fp1);
fread(emp3->name, size, 1, fp1);
printf("\n%d %s",emp3->id,emp3->name);
i++;
}
fclose(fp1);
free(emp3->name);
free(emp3);
}
void update()
{
int id, flag = 0, i = 1;
char s[size];
if (fp1 == NULL)
{
printf("File cant be opened");
return;
}
printf("Enter employee id to update : ");
scanf("%d", &id);
emp3 = (struct emp *)malloc(1*sizeof(struct emp));
emp3->name=(char *)malloc(size*sizeof(char));
while(i<=count)
{
fread(&emp3->id, sizeof(emp3->id), 1, fp1);
fread(emp3->name,size,1,fp1);
if (id == emp3->id)
{
printf("Enter new name of emplyee to update : ");
scanf(" %[^\n]s", s);
fseek(fp1, -204L, SEEK_CUR);
fwrite(&emp3->id, sizeof(emp3->id), 1, fp1);
fwrite(s, size, 1, fp1);
flag = 1;
break;
}
i++;
}
if (flag != 1)
{
printf("No employee record found");
flag = 0;
}
fclose(fp1);
free(emp3->name); /* to free allocated memory */
free(emp3);
}
您在评论中提到 fp
为 NULL。这意味着您实际上没有要写入的打开文件;较早的 fopen
调用 returned NULL 以指示错误。崩溃的 直接 原因是 fwrite
通过尝试访问它应该指向的 FILE
结构的字段来取消引用空指针; 间接 原因是您在使用前没有检查 fopen
中的 return 值。
在您的每个 fopen
调用之后,添加如下内容:
if (fp == NULL)
{
perror("Failed to open file");
exit(1);
}
当 fopen
失败时,它 return 为 NULL 并设置全局变量 errno
以指示出了什么问题。 perror
函数打印与 errno
值相对应的描述性消息。
由于您的程序希望在命令行中获得一个文件名(即在 argv[1]
中,请确保 Xcode 实际上给了它一个文件名。从 IDE 启动程序默认情况下通常在没有任何命令行参数的情况下运行它。当 argc
小于 2 时访问 argv[1]
将是另一个错误,并且可能会显示为不是有效文件名的垃圾字符串,导致fopen
失败。