与Berkeley DB混合使用C ++ / CLI代码

| 我打算在
/clr
模式下以C ++ / CLI使用Berkeley DB。我写了这段代码: 编辑:
// DB_test1.cpp : main project file.

#include \"stdafx.h\"
#pragma comment(lib,\"libdb51\")
using namespace System;
using namespace System::Runtime::InteropServices;

int main(array<System::String ^> ^args)
{
    Db SigDb(0,0);
    unsigned int oFlags= DB_CREATE;
    SigDb.open(NULL,\"SigDb.db\",0,DB_BTREE,oFlags,0);
    String^ HexSig=\"D8B1048900ABFF8B\";
    wchar_t* a=( wchar_t* )Marshal::StringToHGlobalUni(HexSig).ToPointer() ;
    wchar_t* A=( wchar_t* )Marshal::StringToHGlobalUni(HexSig).ToPointer();;

    Dbt key1(&a,100);
    Dbt data1(&A,100);

    Marshal::FreeHGlobal(IntPtr(A));
    int ret= SigDb.put(NULL,&key1,&data1, DB_NOOVERWRITE);
    if(ret==DB_KEYEXIST){
        Console::WriteLine(\"You are trying to insert an exist key!\");
    }


    wchar_t DDData[200];
    Dbt getKey, getData;
    getKey.set_data(&a);

    getKey.set_size(100);
    getData.set_data(DDData);
    getData.set_ulen(200);
    getData.set_flags(DB_DBT_USERMEM);
    Marshal::FreeHGlobal(IntPtr(a));
    if(SigDb.get(NULL,&getKey,&getData,0)==DB_NOTFOUND)
        Console::WriteLine(\"Not Found !\");
    else
        Console::WriteLine(\" {0}\",Marshal::PtrToStringUni((IntPtr)DDData));


    return 0;
}
该代码已成功编译,但显示错误的输出。我只是在托盘中将
String^ HexSig=\"D8B1048900ABFF8B\";
存储在
SigDb.db
中,然后直接读取相同的字符串并打印!结果看起来不像预期的那样像“ 4”,而是一个随机字符串。有任何想法吗? 编辑后: 此段代码始终执行
Console::WriteLine(\"Not Found !\");
已邀请:
我可以看到您的应用有两个问题: 1)在使用缓冲区的内容之前,对Marshal :: FreeHGlobal进行了两次调用。在执行put操作之前,您不应该释放\'A \';在执行put和get操作之后,才应该释放\'a \'。 2)您将指针存储在Berkeley DB中,而不是字符串本身。那是由于Dbt构造函数调用。您的申请是: Dbt key1(&a,100); 它应该是: Dbt键1(a,100); 同样,对于getKey.set_data方法-它应该使用指针,而不是对该指针的引用。 一旦对您的应用程序进行了上述更改,它就会按预期运行。 问候, 亚历克斯·戈罗德 Oracle伯克利数据库
您使用Marshal :: StringToHGlobalUni(),转换后的字符串是wchar_t *,而不是char *。带有utf16编码的Unicode代码点的宽字符串。要获取char *,您需要StringToHGlobalAnsi()。 不要认为这是一次有损的转换,十多年来,dbase引擎已启用Unicode。另一个严重的问题是您不释放分配给该字符串的内存,需要在finally块中调用Marshal :: FreeHGlobal()。从技术上讲,您还应该使用GlobalLock()将返回的HGLOBAL转换为指针,请考虑使用Marshal :: StringToCoTaskMemXxx。

要回复问题请先登录注册