我需要能够从远程计算机的列表中的特定注册表项,以读出值。 我可以用下面的代码在本地做
using Microsoft.Win32;
RegistryKey rkey = Registry.LocalMachine;
RegistryKey rkeySoftware=rkey.OpenSubKey("Software");
RegistryKey rkeyVendor = rkeySoftware.OpenSubKey("VendorName");
RegistryKey rkeyVersions = rkeyVendor.OpenSubKey("Versions");
String[] ValueNames = rkeyVersions.GetValueNames();
foreach (string name in ValueNames)
{
MessageBox.Show(name + ": " + rkeyVersions.GetValue(name).ToString());
}
但我不知道如何获取远程计算机相同的信息。 我是不是即使使用正确的方法或我应该看WMI或其他什么东西?
您可以通过WMI实现这一点,但我想你也可以通过你当前正在使用相同的机制(即Microsoft.Win32命名空间的类)实现它。
你需要考虑的:
OpenRemoteBaseKey方法
上面的链路给出的例子。 它应该像下面的那样简单:
// Open HKEY_CURRENT_USER\Environment
// on a remote computer.
environmentKey = RegistryKey.OpenRemoteBaseKey(
RegistryHive.CurrentUser, remoteName).OpenSubKey(
"Environment");
但是请注意,这会都会有打开远程注册表项安全隐患,所以你可能需要确保您有相关的安全权限做到这一点。 对于这一点,你要考虑的:
性SecurityPermission
和
RegistryPermission的
班的System.Security.Permissions
命名空间。
我发现我可以为CraigTP表明使用OpenRemoteBaseKey()方法,但它要求我更改DEST计算机上的注册表的权限。
这是我写的代码,一旦工作我改变的权限。
RegistryKey rkey = RegistryKey.OpenRemoteBaseKey(RegistryHive.LocalMachine, "RemoteComputer");
RegistryKey rkeySoftware = rkey.OpenSubKey("Software");
RegistryKey rkeyVendor = rkeySoftware.OpenSubKey("VendorName");
RegistryKey rkeyVersions = rkeyVendor.OpenSubKey("Versions");
String[] ValueNames = rkeyVersions.GetValueNames();
foreach (string name in ValueNames)
{
MessageBox.Show(name + ": " + rkeyVersions.GetValue(name).ToString());
}
我还发现,我能得到使用WMI相同的信息,而无需修改权限。 下面是使用WMI代码。
ManagementScope ms = new ManagementScope();
ms.Path.Server = "flebbe";
ms.Path.NamespacePath = "root\\default";
ms.Options.EnablePrivileges = true;
ms.Connect();
ManagementClass mc = new ManagementClass("stdRegProv");
mc.Scope = ms;
ManagementBaseObject mbo;
mbo = mc.GetMethodParameters("EnumValues");
mbo.SetPropertyValue("sSubKeyName", "SOFTWARE\\VendorName\\Versions");
string[] subkeys = (string[])mc.InvokeMethod("EnumValues", mbo, null).Properties["sNames"].Value;
ManagementBaseObject mboS;
string keyValue;
foreach (string strKey in subkeys)
{
mboS = mc.GetMethodParameters("GetStringValue");
mboS.SetPropertyValue("sSubKeyName", "SOFTWARE\\VendorName\\Versions");
mboS.SetPropertyValue("sValueName", strKey);
keyValue = mc.InvokeMethod("GetStringValue", mboS, null).Properties["sValue"].Value.ToString();
MessageBox.Show(strKey + " : " + keyValue);
}
PS我打电话的循环中向GetStringValue()方法,我知道所有的值都是字符串。 如果有多个数据类型,你将需要阅读从EnumValues方法的类型输出参数的数据类型。
在Win32 API允许通过指定计算机名称RegConnectRegistry 。 我不知道的是,.NET包装暴露出这一点。 您还可以使用WMI你所说的。
在我的博客评论者问我后我的堆栈溢出的解决方案,所以在这儿呢。
如何验证和远程使用C#访问注册表
这是基本相同CraigTP的答案,但是它包含了用于验证远程设备一个很好的类。
而该代码是在生产和测试。
这是我最终去了解决方案:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
// add a reference to Cassia (MIT license)
// https://code.google.com/p/cassia/
using Microsoft.Win32;
namespace RemoteRegistryRead2
{
class Program
{
static void Main(string[] args)
{
String domain = "theDomain";
String user = "theUserName";
String password = "thePassword";
String host = "machine-x11";
using (Cassia.UserImpersonationContext userContext = new Cassia.UserImpersonationContext(domain + "\\" + user, password))
{
string userName = System.Security.Principal.WindowsIdentity.GetCurrent().Name;
System.Console.WriteLine("userName: " + userName);
RegistryKey baseKey = RegistryKey.OpenRemoteBaseKey(RegistryHive.LocalMachine, host);
RegistryKey key = baseKey.OpenSubKey(@"SYSTEM\CurrentControlSet\Control\ComputerName\ComputerName");
String computerName = key.GetValue("ComputerName").ToString();
Console.WriteLine(computerName);
}
}
}
}
奇迹般有效 :]
在通过Windows注册表C#安装的程序简单的例子(远程:OpenRemoteBaseKey)
using Microsoft.Win32;
using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
namespace SoftwareInventory
{
class Program
{
static void Main(string[] args)
{
//!!!!! Must be launched with a domain administrator user!!!!!
Console.ForegroundColor = ConsoleColor.Green;
StringBuilder sbOutFile = new StringBuilder();
Console.WriteLine("DisplayName;IdentifyingNumber");
sbOutFile.AppendLine("Machine;DisplayName;Version");
//Retrieve machine name from the file :File_In/collectionMachines.txt
//string[] lines = new string[] { "NameMachine" };
string[] lines = File.ReadAllLines(@"File_In/collectionMachines.txt");
foreach (var machine in lines)
{
//Retrieve the list of installed programs for each extrapolated machine name
var registry_key = @"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall";
using (Microsoft.Win32.RegistryKey key = RegistryKey.OpenRemoteBaseKey(RegistryHive.LocalMachine, machine).OpenSubKey(registry_key))
{
foreach (string subkey_name in key.GetSubKeyNames())
{
using (RegistryKey subkey = key.OpenSubKey(subkey_name))
{
//Console.WriteLine(subkey.GetValue("DisplayName"));
//Console.WriteLine(subkey.GetValue("IdentifyingNumber"));
if (subkey.GetValue("DisplayName") != null)
{
Console.WriteLine(string.Format("{0};{1};{2}", machine, subkey.GetValue("DisplayName"), subkey.GetValue("Version")));
sbOutFile.AppendLine(string.Format("{0};{1};{2}", machine, subkey.GetValue("DisplayName"), subkey.GetValue("Version")));
}
}
}
}
}
//CSV file creation
var fileOutName = string.Format(@"File_Out\{0}_{1}.csv", "Software_Inventory", DateTime.Now.ToString("yyyy_MM_dd_HH_mmssfff"));
using (var file = new System.IO.StreamWriter(fileOutName))
{
file.WriteLine(sbOutFile.ToString());
}
//Press enter to continue
Console.WriteLine("Press enter to continue !");
Console.ReadLine();
}
}
}