【问题标题】:RegEnumValue only returning LOWORDRegEnumValue 只返回 LOWORD
【发布时间】:2013-07-28 05:23:36
【问题描述】:

我无法理解为什么我的来源只返回注册表 DWORD 的 LOWORD 部分。

以下源代码主要基于this MSDN example。我已经添加了一个部分来输出注册表项的值(数据)。

#include <windows.h>
#include <stdio.h>
#include <tchar.h>

#define MAX_KEY_LENGTH 255
#define MAX_VALUE_NAME 16383

void QueryKey(HKEY hKey) 
{ 
    TCHAR    achClass[MAX_PATH] = TEXT("");  // buffer for class name 
    DWORD    cchClassName = MAX_PATH;  // size of class string 
    DWORD    cSubKeys=0;               // number of subkeys 
    DWORD    cbMaxSubKey;              // longest subkey size 
    DWORD    cchMaxClass;              // longest class string 
    DWORD    cValues;              // number of values for key 
    DWORD    cchMaxValue;          // longest value name 
    DWORD    cbMaxValueData;       // longest value data 
    DWORD    cbSecurityDescriptor; // size of security descriptor 
    FILETIME ftLastWriteTime;      // last write time 

    DWORD i, retCode; 

    TCHAR  achValue[MAX_VALUE_NAME];
    DWORD cchValue = MAX_VALUE_NAME;
    DWORD dataType = 0, dataSize;
    LPBYTE data = (LPBYTE)malloc(512);

    // Get the class name and the value count. 
    retCode = RegQueryInfoKey(
      hKey,                    // key handle 
      achClass,                // buffer for class name 
      &cchClassName,           // size of class string 
      NULL,                    // reserved 
      &cSubKeys,               // number of subkeys 
      &cbMaxSubKey,            // longest subkey size 
      &cchMaxClass,            // longest class string 
      &cValues,                // number of values for this key 
      &cchMaxValue,            // longest value name 
      &cbMaxValueData,         // longest value data 
      &cbSecurityDescriptor,   // security descriptor 
      &ftLastWriteTime);       // last write time 

    // Enumerate the key values. 

    if (cValues) 
    {
        printf( "\nNumber of values: %d\n", cValues);

        for (i=0, retCode=ERROR_SUCCESS; i<cValues; i++) 
        { 
            cchValue = MAX_VALUE_NAME; 
            achValue[0] = '\0'; 
            retCode = RegEnumValue(hKey, i, achValue, &cchValue, 
                             NULL, &dataType, data, &dataSize);

            if (retCode == ERROR_SUCCESS && dataType == REG_DWORD ) 
            {
                _tprintf(TEXT("(%d) %s: 0x%08X\n"), i+1, achValue, (DWORD)*data);
            } 
        }
    }
}

void __cdecl _tmain(void)
{
    HKEY hTestKey;

    if (RegOpenKeyEx(HKEY_CURRENT_USER, TEXT("Console"), 0, KEY_READ, &hTestKey) == ERROR_SUCCESS)
    {
        QueryKey(hTestKey);
    }

    RegCloseKey(hTestKey);
}

源按预期输出键值对,但对于某些键(例如 ScreenBufferSize 和 WindowSize)仅打印 LOWORD 部分。

但是当测试代码运行时,一切都很完美。

#include <windows.h>
#include <stdio.h>

int main(void)
{
    DWORD d = 0x00190050;

    printf("DWORD: %08x\n", d);
    printf("HIWORD: %04x, LOWORD: %04x\n", HIWORD(d), LOWORD(d));

    return 0;
}

如何打印完整的 DWORD 值?

【问题讨论】:

    标签: c winapi registry dword


    【解决方案1】:

    您需要在每次调用RegEnumValue() 之前将dataSize 设置为data 的大小,例如:

    dataSize = 512; // size of data
    retCode = RegEnumValue(hKey, i, achValue, &cchValue, NULL, &dataType, data, &dataSize);
    

    MSDN documentation中查看lpcbData的描述

    【讨论】:

    • 嗨,爱德华 - 很好。设置datasize 需要在for 循环的每次迭代中进行,对吗?
    • 是的,因为每次调用都会更新dataSize 与放入data 的实际数据的大小
    【解决方案2】:
    LPBYTE data;
    ...
    (DWORD)*data
    

    它执行以下操作:它从 data 获取 BYTE,然后将其扩展到 DWORD

    相反,您想从内存位置获取DWORD

    * ((DWORD*) data)
    

    【讨论】:

    • 您好,您能解释一下(DWORD)*data*((DWORD*)data) 之间的区别吗?第二个是指向存储在该内存位置的值的指针。但是(DWORD*)data 有何影响?
    • (DWORD*) data 将指针从“指向 8 位值的指针”转换为“指向 32 位值的指针”。然后你使用这个指针来获取它所引用的 32 位值。
    • 还要注意另一个答案——这是您将面临的下一个问题。
    猜你喜欢
    • 2019-02-25
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-11-17
    • 1970-01-01
    • 2021-11-26
    • 1970-01-01
    相关资源
    最近更新 更多