来自调试器的段错误

Segfaul out from debugger

这个模拟数据库的程序有一个大问题,每次我调用 Database_set 函数时,它都会 return 段错误,但是当我使用 gdb 进行调试时,一切都运行良好。

通过更多测试,我发现如果我创建一个新数据库(用于存储数据的文件)而不重用已经创建的数据库,在调试期间一切正常,如果我以这种方式做事,调用 Database_set 不会 return 段错误。

在调试器之外,这个技巧不起作用,每次我尝试访问第 122 行 addr->set 时都会出现段错误,即使 addr 不是 NULL指针,因为之前对 malloc 的调用没有出错。

这是代码:

#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>

struct Address {
    int id;
    int set;
    char *name;
    char *email;
};

struct Database {
    int MAX_DATA;
    int MAX_ROWS;
    struct Address *rows;
};

struct Connection {
    FILE *file;
    struct Database *db;
};

void die(const char *message)
{
    if(errno) {
        perror(message);
    } else {
        printf("ERROR: %s\n", message);
    }

    exit(1);
}

void Address_print(struct Address *addr)
{
    printf("%d %s %s\n",
            addr->id, addr->name, addr->email);
}


void Database_load(struct Connection *conn)
{
    int rc = fread(conn->db, sizeof(struct Database), 1, conn->file);
    if(rc != 1) die("Failed to load database.");
}


struct Connection *Database_open(const char *filename, char mode)
{
    struct Connection *conn = malloc(sizeof(struct Connection));
    if(!conn) die("Memory error");

    conn->db = malloc(sizeof(struct Database));
    if(!conn->db) die("Memory error");

    if(mode == 'c') {
        conn->file = fopen(filename, "w");
    } else {
        conn->file = fopen(filename, "r+");

        if(conn->file) {
            Database_load(conn);
        }
    }

    if(!conn->file) die("Failed to open the file");

    return conn;
}

void Database_close(struct Connection *conn)
{
    if(conn) {
        if(conn->file) fclose(conn->file);
        if(conn->db) free(conn->db);
        free(conn);
    }
}

void Database_write(struct Connection *conn)
{
    rewind(conn->file);

    int rc = fwrite(conn->db, sizeof(struct Database), 1, conn->file);
    if(rc != 1) die("Failed to write database.");

    rc =fflush(conn->file);
    if(rc == -1) die("Cannot flush database.");
}

void Database_create(struct Connection *conn)
{
    printf("Please, insert max database rows limit .....");
    scanf("%d",&conn->db->MAX_ROWS);

    int i = 0;

    conn->db->rows = malloc(sizeof(struct Address)*conn->db->MAX_ROWS);
    if (!conn->db->rows) die("Memory error !");

    for (i = 0; i < conn->db->MAX_ROWS; i++) {

        /* struct Address addr = {.id = i, .set = 0};
        conn->db->rows[i] = addr; */

        conn->db->rows[i].id = 1;
        conn->db->rows[i].set = 0;
    }
}

void Database_set(struct Connection *conn, int id, const char *name, const char *email)
{

    printf("Please, insert max data limit ....");
     scanf("%d", &conn->db->MAX_DATA);


    struct Address *addr = &conn->db->rows[id];

    if(addr->set) die("Already set, delete it first");

    addr->set = 1;

    addr->name = malloc(conn->db->MAX_DATA);
    if(!addr->name) die("Memory error !");

    char *res = strcpy(addr->name, name);
    if(!res) die("Name copy failed");


    addr->email =malloc(conn->db->MAX_DATA);
    if(!addr->email) die("Memory error !");

    res = strcpy(addr->email, email);
    if(!res) die("Email copy failed");


}

void Database_get(struct Connection *conn, int id)
{
    struct Address *addr = &conn->db->rows[id];

    if(addr->set) {
        Address_print(addr);
    } else {
        die("ID is not set");
    }
}

void Database_delete(struct Connection *conn, int id)
{
    conn->db->rows[id].id = id;
    conn->db->rows[id].set = 0;
}

void Database_list(struct Connection *conn)
{
    int i = 0;
    struct Database *db = conn->db;

    for(i = 0; i < conn->db->MAX_ROWS; i++) {
        struct Address *cur = &db->rows[i];

        if(cur->set) {
            Address_print(cur);
        }
    }
}

int main(int argc, char *argv[])
{

    if(argc < 3) die("USAGE: ex17 <dbfile> <action> [action params]");

    char *filename = argv[1];
    char action = argv[2][0];

    struct Connection *conn = Database_open(filename, action);

    int id = 0;

    if (action != 'c') 
    {
        if (argc > 3) id = atoi(argv[4]);
        if (id >= conn->db->MAX_ROWS) die("There's not that many records.");
    }

    switch(action) {
        case 'c':
            Database_create(conn);
            Database_write(conn);
            break;

        case 'g':
            if(argc != 4) die("Need an id to get");
            Database_get(conn, id);
            break;

        case 's':
            if(argc != 6) die("Need id, name, email to set");

            Database_set(conn, id, argv[4], argv[5]);
            Database_write(conn);
            break;

        case 'd':
            if(argc != 4) die("Need id to delete");

            Database_delete(conn, id);
            Database_write(conn);
            break;

        case 'l':
            Database_list(conn);
            break;
        default:
            die("Invalid action, only: c=create, g=get, s=set, d=del, l=list");
    }

    Database_close(conn);

    return 0;
}

我认为你在这里的基本错误是你假设你可以通过这样做 write/read 你的 "database":

    int rc = fwrite(conn->db, sizeof(struct Database), 1, conn->file);

因为你的数据库结构也包含这个字段:

struct Address *rows;

这样是不行的。这会将一个指针(类似于 0xc2b82200)写入您的文件,而不是您的数组本身。当你尝试加载你的文件时,你将指针设置回相同的值,但由于这是新的部分,这现在可能指向一些未分配的内存页面,这给你一个段错误。

您需要逐行写下您的"table"。