RegCreateKeyEx 和 RegOpenKeyEx 成功但 RegSetValueEx 无法设置值

Posted

技术标签:

【中文标题】RegCreateKeyEx 和 RegOpenKeyEx 成功但 RegSetValueEx 无法设置值【英文标题】:RegCreateKeyEx and RegOpenKeyEx succeeds but RegSetValueEx fails to set value 【发布时间】:2016-04-27 04:48:26 【问题描述】:

我正在尝试为 HKCR 下的测试密钥设置值。 RegCreateKeyEx 成功创建测试密钥,RegOpenKeyEx 成功打开密钥。 RegSetValueEx 输出 ERROR_SUCCESS 但实际上并没有写入任何值。我已经解决了之前提出的所有类似问题,建议是在HKLM\SOFTWARE\Wow6432Node 下创建密钥。我在 64 位 win 10 上,testkey 在 HKCR 下; HKLM\SOFTWARE\Wow6432Node 下没有类似的东西。我尝试了两种不同的RegSetValueEx,但都不起作用。我究竟做错了什么?

HKEY hKey;
LONG lResult;
lResult = RegCreateKeyEx(HKEY_CLASSES_ROOT, "testKey", 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL );
if(lResult == ERROR_SUCCESS)
    cout<<"Success! Key Created!"<<endl;

lResult = RegOpenKeyEx(HKEY_CLASSES_ROOT, "testKey", 0, KEY_ALL_ACCESS, &hKey);
if(lResult == ERROR_SUCCESS)
    cout<<"Success! Key Opened!"<<endl;
    const char info[] = "URL:URITest Link\0";
    lResult = RegSetValueEx(HKEY_CLASSES_ROOT,"",0,REG_SZ,(BYTE*) info,strlen(info));
    //lResult = RegSetValueEx(HKEY_CLASSES_ROOT,"URL Protocol",0,REG_NONE,NULL,0);
    if(lResult == ERROR_SUCCESS)
        cout<<"Success! Value Set!"<<endl;
    else
        cout<<lResult<<endl;


编辑:请参阅下面的@Remy Lebeau 答案,了解我的代码哪里出错了。

【问题讨论】:

ERROR_SUCCESS 正如我在 OP 中指出的那样 您使用 KEY_READ 打开了 阅读 密钥。然后你尝试一个值。但是值类型是none,你成功创建了没有值。当然,这些都没有任何意义。 @HansPassant:代码未在使用 KEY_READ` 访问权限打开的 HKEY 上调用 RegSetValueEx()。它正在尝试直接写入 HKEY_CLASSES_ROOT 密钥(这是一个错误),但可能 API 正在接受它。 我需要在testKey下添加两个字符串,默认值需要URL:URITest Link,并且需要有一个名为URL Protocol的空字符串。这是基于Custom URI Scheme for windows @Remy Lebeau 如果你说的是真的,那我怎样才能直接访问testKey 【参考方案1】:

我正在尝试为 HKCR 下的测试密钥设置值。 RegCreateKeyEx 成功创建测试密钥,RegOpenKeyEx 成功打开密钥。 RegSetValueEx 输出 ERROR_SUCCESS 但实际上并没有写入任何值。

您没有将RegOpenKeyEx() 返回的HKEY 句柄传递给RegSetValueEx()。需要将第一个参数中硬编码的HKEY_CLASSES_ROOT替换为hKey变量:

lResult = RegSetValueEx(hKey, "URL Protocol", 0, REG_NONE, NULL,0);

但是,为了使其工作,您还需要更改您的 RegOpenKeyEx() 调用以请求 KEY_WRITE(或至少 KEY_SET_VALUE)权限。您正在使用 KEY_READ 权限打开密钥。您不能写入只读密钥:

lResult = RegOpenKeyEx(HKEY_CLASSES_ROOT, "testKey", 0, KEY_SET_VALUE, &hKey);

话虽如此,您不应该一开始就直接写信给KEY_CLASSES_ROOTHKEY_CLASSES_ROOT 不是它自己的实际键,它实际上是 HKEY_LOCAL_MACHINE\Software\ClassesHKEY_CURRENT_USER\Software\Classes 键的合并视图:

HKEY_CLASSES_ROOT Key

Merged View of HKEY_CLASSES_ROOT.aspx

如果您将密钥写入 HKEY_CLASSES_ROOT 下的密钥,系统会将信息存储在 HKEY_LOCAL_MACHINE\Software\Classes 下。如果您将值写入 HKEY_CLASSES_ROOT 下的某个键,并且该键已经存在于 HKEY_CURRENT_USER\Software\Classes 下,系统会将信息存储在那里而不是 HKEY_LOCAL_MACHINE\Software\Classes 下。

如果你想注册一个只有当前用户可以访问的URI方案,保存在HKEY_CURRENT_USER\Software\Classes\testKey

HKEY hKey;
LONG lResult;

lResult = RegCreateKeyEx(HKEY_CURRENT_USER, "Software\\Classes\\testKey", 0, NULL, REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, &hKey, NULL);
if (lResult == ERROR_SUCCESS)

    cout << "Success! Key Created!" << endl;

    const char info[] = "URL:URITest Link\0";
    lResult = RegSetValueEx(hKey, "" , 0, REG_SZ, (BYTE*) info, sizeof(info));
    if (lResult == ERROR_SUCCESS)
        lResult = RegSetValueEx(hKey, "URL Protocol", 0, REG_NONE, NULL, 0);
    if (lResult == ERROR_SUCCESS)
        cout << "Success! Value Set!" << endl;
    else
        cout << lResult << endl;

    RegCloseKey(hKey);

如果您希望所有用户都可以访问 URI 方案,请将其保存在 HKEY_LOCAL_MACHINE\Software\Classes\testKey

HKEY hKey;
LONG lResult;

lResult = RegCreateKeyEx(HKEY_LOCAL_MACHINE, "Software\\Classes\\testKey", 0, NULL, REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, NULL, &hKey, NULL);
if (lResult == ERROR_SUCCESS)

    cout << "Success! Key Created!" << endl;

    const char info[] = "URL:URITest Link\0";
    lResult = RegSetValueEx(hKey, "" , 0, REG_SZ, (BYTE*) info, sizeof(info));
    if (lResult == ERROR_SUCCESS)
        lResult = RegSetValueEx(hKey, "URL Protocol", 0, REG_NONE, NULL, 0);
    if (lResult == ERROR_SUCCESS)
        cout << "Success! Value Set!" << endl;
    else
        cout << lResult << endl;

    RegCloseKey(hKey);

【讨论】:

@Icarus:是的。此外,在保存您的值时,没有理由调用 RegCreateKeyEx()RegOpenKeyEx() 以获得相同的密钥(特别是因为您从 RegCreateKeyEx() 泄漏了 HKEY)。只需使用RegCreateKeyEx() 返回的HKEY 并将其传递给RegSetValueEx()。创建已经存在的密钥不是错误。我已更新我的答案以显示这些详细信息。 非常感谢您的帮助。我现在明白我错在哪里并修复了代码。【参考方案2】:

我猜你应该将hKey 传递给RegSetValueEx

【讨论】:

并打开它进行写访问。 我试过了。我收到错误消息 2。我认为这意味着该函数无法访问密钥。 错误2是ERROR_FILE_NOT_FOUND,表示找不到请求的key。

以上是关于RegCreateKeyEx 和 RegOpenKeyEx 成功但 RegSetValueEx 无法设置值的主要内容,如果未能解决你的问题,请参考以下文章

C++ - RegCreateKeyEx 成功但没有结果

> 使用 RegCreateKeyEx() 的 Windows 可执行文件不提示输入 UAC 管理员权限并失败? [复制]

如何修复 RegCreateKeyEx() 返回错误 5,“访问被拒绝。”在 SQL Server 2017 中

关于RegCreateKeyEx函数的用法(+50分)

注册表API

关于jmeter运行提示没有权限 报错