无法从 C 中的 Berkeley DB 检索值

Posted

技术标签:

【中文标题】无法从 C 中的 Berkeley DB 检索值【英文标题】:Not able to retrieve values from Berkeley DB in C 【发布时间】:2011-11-11 08:49:44 【问题描述】:

我有两个单独的代码,一个用于插入数据,另一个用于使用 C 在 Berkeley DB 中检索数据。我的问题是我能够看到我的数据已存储,当我执行 db->get 后立即db->put,我看到了值。但是当我尝试通过单独的代码执行 db->get 时,我遇到了问题,当我不使用任何删除功能时,我得到的错误是 db->get: DB_NOTFOUND。不知道哪里出错了

插入数据代码:(我将来自其他函数的字符串传递给它)

int db_json(char *json) 

    typedef struct                              
        char data1[500];    
     pearson_record;

    pearson_record s;
    int i =0;
    DB *dbp;
    DBT key, data;
    int ret, t_ret;
    int recno;

    if ((ret = db_create(&dbp, NULL, 0)) != 0) 
        fprintf(stderr, "db_create: %s\n", db_strerror(ret));
        exit (1);
    

    if ((ret = dbp->open(dbp, NULL, DATABASE, NULL, DB_BTREE, DB_CREATE, 0664)) != 0) 
        dbp->err(dbp, ret, "%s", DATABASE);
        goto err;
    

    printf("data: %s\n",json);
    strncpy(s.data1, json, strlen(json)+1);

    recno = 1;

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

    key.data = &recno;
    key.size = sizeof(recno);
    data.data = &s;
    data.size = sizeof(s);

    if ((ret = dbp->put(dbp, NULL, &key,&data,0)) == 0)
        printf("db: %d: key stored.\n", *(int *)key.data);
    else
    
        dbp->err(dbp, ret, "DB->put");
        goto err;
    

    err:    
        if ((t_ret = dbp->close(dbp, 0)) != 0 && ret == 0)
            ret = t_ret; 

    return 0;

检索数据代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <json/json.h>
#include <curl/curl.h>
#include <sys/types.h>
#include <db.h>

#define  DATABASE "mydata.db"

int main()

    typedef struct                             
        char data1[500];
     pearson_record;

    pearson_record s;
    int i =0;
    DB *dbp;
    DBT key, data;
    int ret, t_ret;
    int recno;               

    if ((ret = db_create(&dbp, NULL, 0)) != 0) 
        fprintf(stderr, "db_create: %s\n", db_strerror(ret));
        exit (1);
       

    if ((ret = dbp->open(dbp, NULL, DATABASE, NULL, DB_BTREE, DB_CREATE, 0664)) != 0) 
        dbp->err(dbp, ret, "%s", DATABASE);
        goto err;
    

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

    pearson_record *ppr;

    if ((ret = dbp->get(dbp, NULL, &key, &data, 0)) == 0)    
        ppr = (pearson_record *) data.data;
        printf("db: %d: key retrieved: data was %s %d\n",
             *(int *)key.data, ppr->data1, data.size);
     
     else 
         dbp->err(dbp, ret, "DB->get");         
    
    goto err;

    err:    
        if ((t_ret = dbp->close(dbp, 0)) != 0 && ret == 0)
            ret = t_ret; 

    return 0;

【问题讨论】:

我刚刚调整了代码块的格式,以便您阅读。不过,这确实是令人不快的代码:为什么您typedef 两次(甚至在您的函数内)使用相同的结构,而不是在两个地方都包含的一个头文件中?在这种情况下没有必要使用goto 你验证过数据真的写入数据库了吗?使用 db_dump 工具? 【参考方案1】:

要使用 DB->get(),您需要知道要检索的记录的确切键。在您的示例中,您必须设置 recno = 1,然后在调用 DB->get() 之前设置密钥 DBT 以指向它。

在您不知道密钥的可能情况下,您想要的是 DB->cursor() 和 DBcursor->get()。您可以使用 DB->cursor() 打开数据库的游标,然后循环调用 DBcursor->get(curs, &key, &data, DB_NEXT)。这将返回数据库中的每条记录,从头开始。

这里的 BDB 程序员参考中的“使用游标检索记录”部分有一个很好的示例:

http://docs.oracle.com/cd/E17076_02/html/programmer_reference/am_cursor.html#am_curget

【讨论】:

以上是关于无法从 C 中的 Berkeley DB 检索值的主要内容,如果未能解决你的问题,请参考以下文章

Berkeley DB 无法在不关闭数据库的情况下访问数据

从值中检索一系列数据 Berkeley DB

Berkeley DB:如何从 QUEUE 获取特定的 KEY

使用 Berkeley DB、C++ STL 接口进行批量读取

Berkeley DB 中的函数指针迭代器

如何使用berkeley db检索键小于指定键的数据?