我正在通过 Learn C The Hard Way 来提高我的 C 技能,目前我在第 17 次练习中。
我正在做“额外学分”部分。使该页面上给出的数据库代码得到我正在尝试
“更改代码以接受 MAX_DATA 和 MAX_ROWS 的参数,将它们存储在 Database 结构中,并将其写入文件,从而创建一个可以任意大小的数据库”
因此,我注释掉了 #define 指令,并按照给定的方式更改了 Address 和 Database 结构:
struct Address {
int id;
int set;
char *name;
char *email;
};
struct Database {
int MAX_DATA;
int MAX_ROWS;
struct Address *rows;
};
这样,我可以从用户那里获取 MAX_DATA 和 MAX_ROWS 参数,以根据他的喜好创建数据库。我在代码中更改的其他内容是——
Database_create 函数:
void Database_create(struct Connection *conn, int MAX_DATA, int MAX_ROWS)
{
int i = 0;
conn->db->MAX_DATA = MAX_DATA;
conn->db->MAX_ROWS = MAX_ROWS;
conn->db->rows = malloc(sizeof(struct Address) * MAX_ROWS);
for(i = 0; i < MAX_ROWS; i++) {
struct Address addr = {.id = i, .set = 0};
conn->db->rows[i] = addr;
}
}
我得到“无效读取 4 个字节”的部分,SegFault 是:
void Database_set(struct Connection *conn, int id, const char *name, const char *email)
{
struct Address *addr = &conn->db->rows[id];
int MAX_DATA = conn->db->MAX_DATA;
if(addr->set) die("Already set, delete it first");
/* This if statement gives error for addr->set */
addr->set = 1;
addr->name = malloc(sizeof(char) * MAX_DATA);
addr->email = malloc(sizeof(char) * MAX_DATA);
// WARNING: bug, read the "How To Break It" and fix this
char *res = strncpy(addr->name, name, MAX_DATA);
// demonstrate the strncpy bug
if(!res) die("Name copy failed");
res = strncpy(addr->email, email, MAX_DATA);
if(!res) die("Email copy failed");
}
我知道这段代码片段不是全部,但我不能在这里粘贴整个代码。所以,我把它贴在这里: http: //pastebin.com/EbKShT3r 我可以使用“c”选项为第一次运行创建和编写数据库。但是,要使用 's' 选项添加条目,则会发生分段错误。
编辑:所以,最后,我让这个程序使用@WhizCraig给出的解决方案但是,为了释放内存,这是我正在尝试的:
void Database_close(struct Connection *conn)
{
int i;
if (conn) {
int MAX_ROWS = conn->db->MAX_ROWS;
for (i=0; i<MAX_ROWS; i++) {
struct Address *row = conn->db->rows+i;
if (row->set) {
free(row->name);
free(row->email);
}
}
free(conn->db->rows);
if(conn->file) fclose(conn->file);
if(conn->db) free(conn->db);
free(conn);
}
}
而且,我在 Valgrind 中遇到错误,内存泄漏。我不明白上面代码中的错误。但是,主要问题似乎已解决:)