伯克利数据库不能使用不同的 Dbt 初始化来设置值
Berkeley DB cannot set value with different Dbt initialization
代码的唯一区别是数据的初始化。这有效:
Dbt key, data(&b, sizeof(int));
key.set_data(&a);
key.set_ulen(sizeof(int));
data.set_flags(DB_DBT_USERMEM);
但这不会:
Dbt key, data;
key.set_data(&a);
key.set_ulen(sizeof(int));
data.set_data(&b);
data.set_ulen(sizeof(int));
data.set_flags(DB_DBT_USERMEM);
我用 https://docs.oracle.com/cd/E17076_04/html/api_reference/CXX/dbt.html 阅读文档,我没有看到这两种方式的区别。这令人困惑。
完整的代码和结果如下所示:
$ cat db.cpp
#include <db.h>
#include <db_cxx.h>
#include <exception>
#include <iostream>
using namespace std;
int main() {
Db db(NULL, 0);
u_int32_t oFlags = DB_CREATE | DB_TRUNCATE;
try {
db.open(NULL, "test.db", NULL, DB_HASH, oFlags, 0);
} catch (DbException &e) {
cout << "DbException" << endl;
} catch (std::exception &e) {
}
int a = 5, b = 6, c = 0, result[1]= {-1};
Dbt key, data(&b, sizeof(int));
key.set_data(&a);
key.set_ulen(sizeof(int));
data.set_flags(DB_DBT_USERMEM);
cout << (db.get(NULL, &key, &data, 0) == DB_NOTFOUND) << endl;
cout << c << endl;
db.put(NULL, &key, &data, 0);
key.set_data(&a);
data.set_data(result);
data.set_ulen(sizeof(int));
cout << (db.get(NULL, &key, &data, 0) )<< endl;
cout << *((int *) data.get_data()) << endl;
cout << result[0] << endl;
return 0;
}
$ rm test.db ; g++ db.cpp -ldb_cxx-5.1; ./a.out
1
0
0
6
6
$ cat db.cpp
#include <db.h>
#include <db_cxx.h>
#include <exception>
#include <iostream>
using namespace std;
int main() {
Db db(NULL, 0);
u_int32_t oFlags = DB_CREATE | DB_TRUNCATE;
try {
db.open(NULL, "test.db", NULL, DB_HASH, oFlags, 0);
} catch (DbException &e) {
cout << "DbException" << endl;
} catch (std::exception &e) {
}
int a = 5, b = 6, c = 0, result[1]= {-1};
Dbt key, data;
key.set_data(&a);
key.set_ulen(sizeof(int));
data.set_data(&b);
data.set_ulen(sizeof(int));
data.set_flags(DB_DBT_USERMEM);
cout << (db.get(NULL, &key, &data, 0) == DB_NOTFOUND) << endl;
cout << c << endl;
db.put(NULL, &key, &data, 0);
key.set_data(&a);
data.set_data(result);
data.set_ulen(sizeof(int));
cout << (db.get(NULL, &key, &data, 0) )<< endl;
cout << *((int *) data.get_data()) << endl;
cout << result[0] << endl;
return 0;
}
$ rm test.db ; g++ db.cpp -ldb_cxx-5.1; ./a.out
1
0
0
-1
-1
你需要调用Dbt.set_size()
。 Dbt.set_ulen()
仅设置 Dbt.data
指向的内存的分配大小。 Dbt.size
具有密钥或数据的实际使用长度。 将这些调用添加到第二个示例中应该可以使其正常工作:
Dbt key, data;
key.set_data(&a);
key.set_ulen(sizeof(int));
key.set_size(sizeof(int));
data.set_data(&b);
data.set_ulen(sizeof(int));
data.set_size(sizeof(int));
data.set_flags(DB_DBT_USERMEM);
至于为什么Dbt key
在没有set_size
的情况下为你工作,我完全不确定。 也许您对堆栈上的未初始化数据很幸运? 如果你想知道,你可以打印出key.size
的值。
明确的是,您无需在put()
之前设置ulen
。 它只需要在你做一个get()
之前设置,让BDB知道它必须读入多少内存。
相关文章:
- 是否可以初始化不可复制类型的成员变量(或基类)
- C++使用整数的压缩数组初始化对象
- C++初始化基类
- 多成员Constexpr结构初始化
- 复制列表初始化的隐式转换的等级是多少
- 内联映射初始化的动态atexit析构函数崩溃
- 如何在C++中初始化嵌套类中的2个memeber
- 如何声明特征矩阵,然后通过嵌套循环初始化它
- 没有用于初始化C++中的变量模板的匹配构造函数
- 在未初始化映射的情况下,将值插入到映射的映射中
- C++成员初始化
- 为什么在C++中首先初始化成员类
- 同时具有"聚合初始化"和"模板推导"
- 初始化具有非默认构造函数的std::数组项的更好方法
- 是否可以在编译时初始化数组,以便在运行时不会花费时间?
- 我可以使用条件运算符初始化C风格的字符串文字吗
- 在C和C++中初始化结构中的数组
- 标准是否使用多余的大括号(例如 T{{{10}}})定义列表初始化?
- 在函数内部的声明中初始化数组,并在外部使用它
- 伯克利数据库不能使用不同的 Dbt 初始化来设置值