RegQueryValueEx - 向此函数添加什么代码以显示ERROR_SUCCESS

时间:2009-10-03 16:46:06

标签: c++ winapi registry

什么代码添加到此功能才能正常工作? (ERROR_SUCCESS)

我有代码,检查注册表中的值。

在函数RegQueryValueEx中是bug。 当oldValue字母长于newValue时,函数会显示ERROR_MORE_DATA,但我想要ERROR_SUCCESS

为此功能添加了哪些代码?

void function(string newValue, string key, string name)

{

 // string key - key in registry, ie Myapp\\Options
 // string name - name in registry
 // string newValue - data in REG_SZ


 string oldValue;
 DWORD keytype = REG_SZ;
    HKEY keyHandle;
 DWORD size = sizeof(string);
 if(RegOpenKeyEx(HKEY_LOCAL_MACHINE, key.c_str(),0L,KEY_ALL_ACCESS,&keyHandle) == ERROR_SUCCESS)
 {


  LONG isgood = RegQueryValueEx(keyHandle, name.c_str(), 0, &keytype, (LPBYTE)&oldValue, &size);
  if(isgood == ERROR_MORE_DATA)
   {
    cout << "Error more data\n"; 
   }
  if(isgood == ERROR_SUCCESS)
  {
   cout << "Old data is " << oldValue.c_str() << endl;
   cout << "New data is " << newValue.c_str() << endl;
   if(strcmp(newValue.c_str(), oldValue.c_str()) != 0) // compare 2 strings, if
   {
    cout << "String 1 and string 2 are different";

   }
   else
   {
    cout << "String 1 and string 2 are the same";
   }
  }
  if(isgood == ERROR_FILE_NOT_FOUND)
  {
   cout << "Name in registry not found!";
  }
 }

}

2 个答案:

答案 0 :(得分:2)

ERROR_MORE_DATA表示您需要传入更大的字符串缓冲区。您需要使用的典型模式是调用一次以获取大小,然后分配适当大小的缓冲区,然后再次调用。或者,你可以猜测一个大小,传入那个大小的缓冲区,如果你得到ERROR_MORE_DATA就增加大小。

顺便说一下,你的计算大小也不正确。而且你没有关闭注册表项。并且您不准备支持在unicode或非unicode模式下编译。

以下是一些解决这些问题的修订代码。

#include <string>
#include <vector>
#include <iostream>

#include <windows.h>
using namespace std;

namespace std
{
#ifdef _UNICODE
    #define tcout wcout
    #define tcin wcin
    typedef wstring tstring;
#else
    #define tcout cout
    #define tcin cin
    typedef string tstring;
#endif
};

void function(tstring newValue, tstring key, tstring name)
{
    // string key - key in registry, ie Myapp\\Options
    // string name - name in registry
    // string newValue - data in REG_SZ
    HKEY keyHandle;
    if(RegOpenKeyEx(HKEY_LOCAL_MACHINE, key.c_str(),0L,KEY_ALL_ACCESS,&keyHandle) == ERROR_SUCCESS)
    {
        DWORD size = 500;   // initial size
        vector<TCHAR> buf(size);
        tstring oldValue;
        DWORD keytype = REG_SZ;

        LONG isgood = RegQueryValueEx(keyHandle, name.c_str(), 0, &keytype, (LPBYTE) &buf[0], &size);
        if(isgood == ERROR_SUCCESS)
        {
            oldValue.assign (&buf[0], size);
        }
        else if(isgood == ERROR_MORE_DATA)
        {
            buf.reserve (size); // expand to however large we need
            isgood = RegQueryValueEx(keyHandle, name.c_str(), 0, &keytype, (LPBYTE)&buf[0], &size);
            if(isgood == ERROR_SUCCESS)
                oldValue.assign (&buf[0], size);
        }
        RegCloseKey (keyHandle);    // remember to close this!
        if(isgood == ERROR_SUCCESS)
        {
            tcout << _T("Old data is ") << oldValue << endl;
            tcout << _T("New data is ") << newValue << endl;
            if(newValue.compare(oldValue) != 0) // compare 2 strings, if
            {
                tcout << _T("String 1 and string 2 are different");

            }
            else
            {
                tcout << _T("String 1 and string 2 are the same");
            }
        }
        if(isgood == ERROR_FILE_NOT_FOUND)
        {
            tcout << _T("Name in registry not found!");
        }
    }
}

int _tmain(int argc, _TCHAR* argv[])
{
    tstring val;
    function (val, _T("SOFTWARE\\Microsoft\\Windows\\CurrentVersion"), _T("CommonFilesDir"));
    return 0;
}

答案 1 :(得分:2)

ERROR_MORE_DATA表示您提供的用于保存数据的缓冲区不够大。

你的问题很多:

  1. 当你说sizeof(字符串)时,你得到的是字符串数据类型的大小,而不是字符串的长度。你应该调用string :: size()来获取字符串的长度。
  2. 您不能只是将字符串转换为LPBYTE。那将是悲惨的失败。 Registry API不适用于string,它们旨在与char*WCHAR*类型配合使用。您需要声明一个本地字符数组(例如char *foo = new char[256]),然后传递它。如果你得到ERROR_MORE_DATA,请声明一个更大的。