我有一个开放的Registry,例如HKEY_CURRENT_USER\SOFTWARE\Microsoft。如何摆脱它的基础注册表键或注册表蜂巢?
RegistryKey.OpenBaseKey(RegistryHive,RegistryView)方法没有将RegistryKey或字符串传递给它的重载。
我编写了以下代码:
// OUR INPUT:
RegistryKey inputKey = Registry.CurrentUser.OpenSubKey("SOFTWARE\\Microsoft");
string subKeyName = subkey.ToString();
string hiveName = subKeyName.Substring(0, subKeyName.IndexOf('\\'));
RegistryHive regHive;
switch (hiveName)
{
case "HKEY_CLASSES_ROOT": regHive = RegistryHive.ClassesRoot; break;
case "HKEY_CURRENT_USER": regHive = RegistryHive.CurrentUser; break;
case "HKEY_LOCAL_MACHINE": regHive = RegistryHive.LocalMachine; break;
case "HKEY_USERS": regHive = RegistryHive.Users; break;
case "PerformanceData": regHive = RegistryHive.LocalMachine; break;
case "CurrentConfig": regHive = RegistryHive.CurrentConfig; break;
case "DynData": regHive = RegistryHive.DynData; break;
default: throw new System.ArgumentOutOfRangeException();
}
// OUR TARGET:
RegistryKey baseKey = RegistryKey.OpenBaseKey(regHive, RegistryView.Default);这是可行的,但在我看来,应该有更好的解决办法。
也许你知道另一个解决办法?
发布于 2019-10-24 19:23:49
@TheGeneral,谢谢你的源代码链接。在registry.cs中,我找到了一种可以满足我需要的方法。不幸的是,它是私有方法,我们需要在我们的项目中再次重写它:
微软解决方案:
// Following function will parse a keyName and returns the basekey for it.
// It will also store the subkey name in the out parameter.
// If the keyName is not valid, we will throw ArgumentException.
// The return value shouldn't be null.
//
[System.Security.SecurityCritical] // auto-generated
private static RegistryKey GetBaseKeyFromKeyName(string keyName, out string subKeyName) {
if( keyName == null) {
throw new ArgumentNullException("keyName");
}
string basekeyName;
int i = keyName.IndexOf('\\');
if( i != -1) {
basekeyName = keyName.Substring(0, i).ToUpper(System.Globalization.CultureInfo.InvariantCulture);
}
else {
basekeyName = keyName.ToUpper(System.Globalization.CultureInfo.InvariantCulture);
}
RegistryKey basekey = null;
switch(basekeyName) {
case "HKEY_CURRENT_USER":
basekey = Registry.CurrentUser;
break;
case "HKEY_LOCAL_MACHINE":
basekey = Registry.LocalMachine;
break;
case "HKEY_CLASSES_ROOT":
basekey = Registry.ClassesRoot;
break;
case "HKEY_USERS":
basekey = Registry.Users;
break;
case "HKEY_PERFORMANCE_DATA":
basekey = Registry.PerformanceData;
break;
case "HKEY_CURRENT_CONFIG":
basekey = Registry.CurrentConfig;
break;
case "HKEY_DYN_DATA":
basekey = RegistryKey.GetBaseKey(RegistryKey.HKEY_DYN_DATA);
break;
default:
throw new ArgumentException(Environment.GetResourceString("Arg_RegInvalidKeyName", "keyName"));
}
if( i == -1 || i == keyName.Length) {
subKeyName = string.Empty;
}
else {
subKeyName = keyName.Substring(i + 1, keyName.Length - i - 1);
}
return basekey;
}我的解决方案:我将它重写为一个扩展方法:
public static RegistryHive GetRegistryHive(this RegistryKey key)
{
if (key == null)
{
throw new System.ArgumentNullException(nameof(key));
}
int i = key.Name.IndexOf('\\');
string basekeyName = (i != -1) ? key.Name.Substring(0, i) : key.Name;
switch (basekeyName)
{
case "HKEY_CLASSES_ROOT": return RegistryHive.ClassesRoot;
case "HKEY_CURRENT_USER": return RegistryHive.CurrentUser;
case "HKEY_LOCAL_MACHINE": return RegistryHive.LocalMachine;
case "HKEY_USERS": return RegistryHive.Users;
case "HKEY_PERFORMANCE_DATA": return RegistryHive.LocalMachine;
case "HKEY_CURRENT_CONFIG": return RegistryHive.CurrentConfig;
case "HKEY_DYN_DATA": return RegistryHive.DynData;
default: throw new System.ArgumentException(nameof(basekeyName));
}
}
public static RegistryKey OpenBaseKey(this RegistryKey key)
{
return RegistryKey.OpenBaseKey(GetRegistryHive(key), key.View);
}发布于 2019-10-22 19:45:13
你基本上是在正确的轨道上,做一切可以做的事情。
注册表类只是win32 API的一个包装器,这里没有真正的魔力,可以在注册表和单元的字符串路径上工作。
更多的链接:
正如你已经注意到的,没有更容易的方法来实现你想要的目标。
https://stackoverflow.com/questions/58510869
复制相似问题