1

所以我目前正在通过 C 语言处理伯克利数据库,我遇到了一个关于允许重复数据的小问题。如果我不允许通过 DB->set_flags 重复条目,并且我使用游标遍历数据库中的项目,那么一切正常。当我在我的程序中添加 DB->set_flags 行并且没有更改任何其他内容时,我遇到了分段错误并且我不知道如何修复它,考虑到它可以正常工作而没有重复条目。代码如下:

DB *dates_db;
db_create(&dates_db, NULL, 0);
dates_db->set_flags(dates_db, DB_DUP);
dates_db->open(dates_db, NULL, "da.idx", NULL, DB_BTREE, 0, 0664);

DBT key, data;
memset(&key, 0, sizeof(key)); 
memset(&data, 0, sizeof(data));

DBC *DBpointer;
dates_db->cursor(dates_db, NULL, &DBpointer, 0);

while(DBpointer->c_get(DBpointer, &key, &data, DB_NEXT) != DB_NOTFOUND)
{
    printf("The key is: %s\nThe data is: %s\n", (char *)key.data, (char *)data.data);
}

谁能告诉我我需要添加什么?我已经检查了包含有关 Berkeley DB 的所有信息的 sourceforge 页面,据我所知,我只需要添加 DB->set_flags 行以允许重复条目,但是我可能错过了什么吗?

4

1 回答 1

1

首先,当函数可能失败时,您应该始终检查返回值。

具体来说,db_create, set_flags, open,cursor 返回错误指示。

如果正如您在评论中指出的那样,它是c_get导致 SIGSEGV 的原因,您可能希望确保光标确实是正确创建的。

改变:

DBC *DBpointer;
dates_db->cursor(dates_db, NULL, &DBpointer, 0);

到:

DBC *DBpointer = (DBC*)0xdeadbeef;
int rc = dates_db->cursor(dates_db, NULL, &DBpointer, 0);
printf ("DEBUG: %d %p\n", rc, DBpointer);
fflush (stdio); // and possibly also: fsync (fileno (stdio));

将是一个好的开始。

最好是一路走来并使用类似的东西:

#define CHKERR(x) if(rc!=0){printf("%s err=%d\n",x,rc);fflush(stdout);exit(1);}
int rc;
DB *dates_db;

rc = db_create(&dates_db, NULL, 0);
CHKERR("create");

rc = dates_db->set_flags(dates_db, DB_DUP);
CHKERR("set_flags");

rc = dates_db->open(dates_db, NULL, "da.idx", NULL, DB_BTREE, 0, 0664);
CHKERR("open");

DBT key, data;
memset(&key, 0, sizeof(key)); 
memset(&data, 0, sizeof(data));

DBC *DBpointer;
rc = dates_db->cursor(dates_db, NULL, &DBpointer, 0);
CHKERR("cursor");

while((rc = DBpointer->c_get(DBpointer, &key, &data, DB_NEXT)) != DB_NOTFOUND)
{
    CHKERR("c_get");
    printf("The key is: %s\nThe data is: %s\n", (char *)key.data, (char *)data.data);
}

根据您在公开呼叫中收到错误 22 的进一步评论,那是EINVAL(至少在我的系统上)这意味着您的参数之一不正确。

根据网络搜索,该调用似乎有一些变化open,范围在 5 到 7 个参数之间。一些Oracle BDB doco声明它需要五个参数(没有数据库指针和事务指针),但同一文档(和其他 Oracle doco)中的示例代码具有您的七参数形式。

查看您的 BDB 头文件以查看您应该使用哪个可能是值得的。

于 2012-04-05T04:42:15.280 回答