【转】使用.netFx4.0提供的方法解决32位程序访问64位系统的64位注册表
我们知道目标平台是32位的程序运行在64位的系统上,去访问部分注册表的时候系统自动重定向到win32node节点对应的项去了。但是做过安装程序开发人员可能遇到过“需要去掉重定向”的问题,即直接访问64位程序的注册表。
网上有很多winAPI的方法,关闭注册表的重定向稍微复杂。(关闭文件系统的重定向稍微简单些,搬过来就可以用;关闭注册表的重定向我现在没看懂。)
我这里提供的方法不需要关闭重定向,也不需要用winAPI,操作起来方便了许多。具体如下:
第一个方法是获得根节点的句柄,常数是固定的。
1 static IntPtr GetHiveHandle(RegistryHive hive) 2 { 3 IntPtr preexistingHandle = IntPtr.Zero; 4 5 IntPtr HKEY_CLASSES_ROOT = new IntPtr(-2147483648); 6 IntPtr HKEY_CURRENT_USER = new IntPtr(-2147483647); 7 IntPtr HKEY_LOCAL_MACHINE = new IntPtr(-2147483646); 8 IntPtr HKEY_USERS = new IntPtr(-2147483645); 9 IntPtr HKEY_PERFORMANCE_DATA = new IntPtr(-2147483644); 10 IntPtr HKEY_CURRENT_CONFIG = new IntPtr(-2147483643); 11 IntPtr HKEY_DYN_DATA = new IntPtr(-2147483642); 12 switch (hive) 13 { 14 case RegistryHive.ClassesRoot: preexistingHandle = HKEY_CLASSES_ROOT; break; 15 case RegistryHive.CurrentUser: preexistingHandle = HKEY_CURRENT_USER; break; 16 case RegistryHive.LocalMachine: preexistingHandle = HKEY_LOCAL_MACHINE; break; 17 case RegistryHive.Users: preexistingHandle = HKEY_USERS; break; 18 case RegistryHive.PerformanceData: preexistingHandle = HKEY_PERFORMANCE_DATA; break; 19 case RegistryHive.CurrentConfig: preexistingHandle = HKEY_CURRENT_CONFIG; break; 20 case RegistryHive.DynData: preexistingHandle = HKEY_DYN_DATA; break; 21 } 22 return preexistingHandle; 23 } 24 25 /// <summary> 26 /// 用于32位程序访问64位注册表 27 /// </summary> 28 /// <param name="hive">根级别的名称</param> 29 /// <param name="keyName">不包括根级别的名称</param> 30 /// <param name="valueName">项名称</param> 31 /// <param name="view">注册表视图</param> 32 /// <returns>值</returns> 33 static object GetValueWithRegView(RegistryHive hive, string keyName, string valueName, RegistryView view) 34 { 35 36 SafeRegistryHandle handle = new SafeRegistryHandle(GetHiveHandle(hive), true);//获得根节点的安全句柄 37 38 RegistryKey subkey = RegistryKey.FromHandle(handle, view).OpenSubKey(keyName);//获得要访问的键 39 40 RegistryKey key = RegistryKey.FromHandle(subkey.Handle, view);//根据键的句柄和视图获得要访问的键 41 42 return key.GetValue(valueName);//获得键下指定项的值 43 } 44 /// <summary> 45 /// 用于32位的程序设置64位的注册表 46 /// </summary> 47 /// <param name="hive">根级别的名称</param> 48 /// <param name="keyName">不包括根级别的名称</param> 49 /// <param name="valueName">项名称</param> 50 /// <param name="value">值</param> 51 /// <param name="kind">值类型</param> 52 /// <param name="view">注册表视图</param> 53 static void SetValueWithRegView(RegistryHive hive, string keyName, string valueName, object value, RegistryValueKind kind, RegistryView view) 54 { 55 SafeRegistryHandle handle = new SafeRegistryHandle(GetHiveHandle(hive), true); 56 57 RegistryKey subkey = RegistryKey.FromHandle(handle, view).OpenSubKey(keyName, true);//需要写的权限,这里的true是关键。0227更新 58 59 RegistryKey key = RegistryKey.FromHandle(subkey.Handle, view); 60 61 key.SetValue(valueName, value, kind); 62 }
使用这些方法需要引入Microsoft.Win32.SafeHandles;Microsoft.Win32;system;这三个命名空间。方法只提供了主干,没有异常处理。望大家及时交流实际使用情况,不足之处请及时指出。
关于winAPI的方法可以访问http://home.cnblogs.com/u/xuguilin/相应的文章,同时也感谢xuguilin对我的指导。据介绍dotNet框架可能需要4.0版本以上