1

我正在尝试在 C++/CLI/clr模式下使用 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然后直接读取相同的字符串并打印它!结果不像D8B1048900ABFF8B预期的那样出现,而是显示为随机字符串。有任何想法吗?

编辑后: 这段代码总是被执行Console::WriteLine("Not Found !");

4

2 回答 2

2

我可以看到您的应用程序存在两个问题:

1) 对 Marshal::FreeHGlobal 的两次调用是在使用缓冲区内容之前进行的。在 put 操作之后,您不应该释放 'A',并且在 put 和 get 操作之后,您不应该释放 'a'。

2)您将指针存储在 Berkeley DB 中,而不是字符串本身。这是由于 Dbt 构造函数调用。您的应用程序是: Dbt key1(&a,100); 应该是:dbt key1(a, 100);

对于 getKey.set_data 方法也是如此——它应该使用指针,而不是对指针的引用。

一旦我对您的应用程序进行了上述更改,它就会按预期运行。

问候, Alex Gorrod Oracle Berkeley DB

于 2011-04-21T00:23:23.017 回答
1

您使用 Marshal::StringToHGlobalUni(),转换后的字符串是 wchar_t*,而不是 char*。具有以 utf16 编码的 Unicode 代码点的宽字符串。要获得 char*,您需要 StringToHGlobalAnsi()。

请考虑这是一个有损转换,dbase 引擎已经启用 Unicode 十多年了。另一个严重的问题是您没有释放为该字符串分配的内存,因此需要在 finally 块中调用 Marshal::FreeHGlobal()。您还应该在技术上使用 GlobalLock() 将返回的 HGLOBAL 转换为指针,考虑 Marshal::StringToCoTaskMemXxx。

于 2011-04-19T12:50:59.577 回答