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 失败。