C# DWORD 和 QWORD - 有符号和无符号的疯狂

Ale*_*x K 5 c# registry dword

我注意到写入注册表时的DWordQWord值应该是有符号整数,而不是无符号整数。如果值为 UInt64 或 UInt32,此代码将引发异常:

registryKey.SetValue(name, value);
Run Code Online (Sandbox Code Playgroud)

根据 MSDNDWORD是一个 32 位无符号整数(范围:0 到 4294967295 十进制)https://msdn.microsoft.com/en-us/library/cc230318.aspx

因此,要将新DWORD值写入注册表,我需要将其转换为有符号整数,如下所示:

UInt32 unsignedValue = (UInt32)someValue;
Int32 signedValue = (Int32)unsignedValue;
registryKey.SetValue(name, signedValue);
Run Code Online (Sandbox Code Playgroud)

将无符号值传递给SetValue方法将引发异常。我是错过了什么还是我只是被智障了?

xan*_*tos 5

由于历史原因,.NET API/库通常是“已签名”而不是“已签名+未签名”。

但最终asigned int和aunsigned int都占用了相同的内存空间,负值没有做特殊处理。所以你可以像你说的那样做:将无符号值转换为有符号值,用它来写SetValue,然后如果你查看其中的值,Regedit你会看到它被写成“无符号”。

请注意,如果您的程序是在“已检查”模式下编译的,则更正确的代码是:

uint unsignedValue = ... // Your original value

int signedValue = unchecked((int)unsignedValue);
registryKey.SetValue(name, signedValue);
Run Code Online (Sandbox Code Playgroud)

因为在“检查”模式下intuint如果转换不可能,则在和之间进行转换可能会引发异常。

注意,写在这里

此 SetValue 重载将 64 位整数存储为字符串 (RegistryValueKind.String)。要将 64 位数字存储为 RegistryValueKind.QWord 值,请使用指定 RegistryValueKind 的 SetValue(String, Object, RegistryValueKind) 重载。

显然,您必须对签名未签名进行相同的处理。