1
如果叫你实现远程启动别人的计算机,你首先想到的可能是先做一个在远程计算机上面运行客户端程序,然后在本地计算机上面再做一个服务器端程序,通过这二个程序直接的通讯实现重启远程计算机。这当然是一个方法。但这未免有点麻烦。如果现在只告诉你远程计算机的管理者的登陆帐号,而并不允许你在远程的计算机上面运行一个所谓的客户端程序,让你通过程序来完成重启远程计算机。不知道你是否感觉有些困难了。其实按照上面的这些条件实现重启远程计算机,利用C#可以比较方便的完成。下面就来介绍一下具体的实现方法。
一. C#重启远程计算机的一些理论知识:
C#实现启动远程计算机的原理是"视窗管理规范"。就是所谓的"WMI"(Windows Management Instrumentation)。Windows 管理规范 (WMI) 支持通过 Internet 管理系统的结构。通过提供管理环境的一致观察,WMI 为用户提供通用访问管理信息。该管理的一致性使您能够管理整个系统,而不只是组件。从 Microsoft MSDN上,您可以获得有关 WMI 软件开发工具包 (SDK) 的详细信息。
WMI(Windows 管理规范)支持有限的安全格式,允许用户在本地计算机或远程计算机上连接 WMI 之前要验证每个用户。这种安全性是操作系统已有的安全顶端的另一层。WMI 不覆盖或破坏由操作系统提供的任何现有的安全性。在默认情况下,管理员组的所有成员都可以完全控制它管理的计算机上的 WMI 服务。其他所有用户在其本地计算机上只有读取/写入/执行的权限。可以通过向被管理的计算机上的管理员组添加用户,或者在 WMI 中授权用户或组并设置权限级别来更改权限。访问基于 WMI 名称空间。在一般情况下,脚本程序的默认命名空间是"root/cimv2"。
在WMI中有着许多足以令我们感觉惊奇的功能。重启远程计算机只是一个很小的功能。在程序中使用WMI可以编写出许多远程管理类型的应用程序。由于在.Net FrameWork SDK中提供了可以直接操作WMI的名称空间,所以C#就可以利用在这些名称空间中定义了的类来充分使用WMI控制给我们带来的各种方便。
二.程序设计和运行的环境设置:
(1).视窗2000服务器版
(2). .Net FrameWork SDK Beta 2
(3).远程计算机的管理者帐号
以上这些不仅是本地计算机配置,还是远程计算机的配置。
三.实现重启远程计算机所使用到在.Net FrameWork SDK Beta 2用以操作WMI名称空间和类:
在.Net FrameWork SDK Beta 2中用来操作WMI的名称空间主要是"System.Management"。要实现重启远程计算机所使用到的类主要有六个:
. "ConnectionOptions"类主要定义远程计算机的管理员帐号;
. "ManagementScope"主要是以给定的管理员帐号连接给定计算机名或者IP地址的计算机;
. "ObjectQuery"类功能是定义对远程计算机要实现那些地远程操作;
. "ManagementObjectSearcher"类从已经完成远程连接的计算机中,得到有那些WMI操作;
. "ManagementObjectCollection"类存放得到WMI操作;
. "ManagementObject"类调用远程计算机可进行WMI操作。
在本文介绍的操作就是重启操作。
四.C#重启远程计算机的重要步骤和实现方法:
(1).连接远程计算机:
按照下列语句可以实现连接远程计算机:
ConnectionOptions options = new ConnectionOptions ( ) ; options.Username ="管理者帐号用户名"; options.Password = "管理者帐号口令" ; ManagementScope scope = new ManagementScope( "" + "远程计算机名或IP地址" + "//root//cimv2", options ) ; //用给定管理者用户名和口令连接远程的计算机 scope.Connect ( ) ; |
(2).得到在远程计算机中可以进行WMI控制:
System.Management.ObjectQuery oq = new System.Management.ObjectQuery ( "SELECT * FROM Win32_OperatingSystem" ) ; ManagementObjectSearcher query1 = new ManagementObjectSearcher ( scope , oq ) ; //得到WMI控制 ManagementObjectCollection queryCollection1 = query1.Get ( ) ; |
(3).调用WMI控制,实现重启远程计算机:
foreach ( ManagementObject mo in queryCollection1 ) { string [ ] ss= { "" } ; //重启远程计算机 mo.InvokeMethod ( "Reboot" , ss ) ; } |
五.C#实现重启远程计算机的源程序代码(boot.cs)和执行界面:
在了解了C#实现重启远程计算机的这些重要步骤后,就可以从容的得到重启远程计算机的完整代码,具体如下:
using System ; using System.Drawing ; using System.Collections ; using System.ComponentModel ; using System.Windows.Forms ; using System.Data ; using System.Management ; public class Form1 : Form { private TextBox textBox1 ; private TextBox textBox2 ; private TextBox textBox3 ; private Label label1 ; private Label label2 ; private Label label3 ; private Button button1 ; private System.ComponentModel.Container components = null ; public Form1 ( ) SuspendLayout ( ) ; textBox2.Location = new System.Drawing.Point ( 138 , 85 ) ; textBox3.Location = new System.Drawing.Point ( 139 , 120 ) ; label1.Location = new System.Drawing.Point ( 24 , 50 ) ; label2.Location = new System.Drawing.Point ( 37 , 88 ) ; label3.Location = new System.Drawing.Point ( 37 , 125 ) ; button1.Location = new System.Drawing.Point ( 95 , 168 ) ; this.AutoScaleBaseSize = new System.Drawing.Size ( 6 , 14 ) ; this.Name = "Form1" ; } |
六.总结:
其实WMI控制可以实现很多以前让我们很头痛的操作。并且使用WMI编写的管理程序也比不用WMI来实现同样功能的程序在设计难度上大大减轻。WMI内容十分丰富,重新启动远程计算机只是其中的一个最为基本的操作。在使用WMI控制之前有一点必须记住,就是你必须知道你所要进行操作的远程计算机的超级管理者的帐号,这是使用WMI的一个前提。
2
远程重启计算机(C#)
首先添加对 System.Management的引用
然后代码入下:
ConnectionOptions op = new ConnectionOptions ( ) ;
op.Username ="wgscd";'或者你的帐号(注意要有管理员的权限)
op.Password = "wgscd" ; '你的密码
ManagementScope scope = new ManagementScope( "" + "106.54.20.1" + "//root//wgscd",
op ) ;
try
{
scope.Connect ( ) ;
System.Management.ObjectQuery oq = new System.Management.ObjectQuery ( "SELECT * FROM
Win32_OperatingSystem" ) ;
ManagementObjectSearcher query1 = new ManagementObjectSearcher ( scope , oq ) ;
//得到WMI控制
ManagementObjectCollection queryCollection1 = query1.Get ( ) ;
foreach ( ManagementObject mobj in queryCollection1 )
{
string [ ] str= { "" } ;
mobj.InvokeMethod ( "Reboot" , str ) ;
}
}
catch
{
}
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
namespace chkipsrv
... {
public class Shudown
...{
[StructLayout(LayoutKind.Sequential, Pack = 1)]
internal struct TokPriv1Luid
...{
public int Count;
public long Luid;
public int Attr;
}
[DllImport("kernel32.dll", ExactSpelling = true)]
internal static extern IntPtr GetCurrentProcess();
[DllImport("advapi32.dll", ExactSpelling = true, SetLastError = true)]
internal static extern bool OpenProcessToken(IntPtr h, int acc, ref IntPtr phtok);
[DllImport("advapi32.dll", SetLastError = true)]
internal static extern bool LookupPrivilegeValue(string host, string name, ref long pluid);
[DllImport("advapi32.dll", ExactSpelling = true, SetLastError = true)]
internal static extern bool AdjustTokenPrivileges(IntPtr htok, bool disall,
ref TokPriv1Luid newst, int len, IntPtr prev, IntPtr relen);
[DllImport("user32.dll", ExactSpelling = true, SetLastError = true)]
internal static extern bool ExitWindowsEx(int DoFlag, int rea);
internal const int SE_PRIVILEGE_ENABLED = 0x00000002;
internal const int TOKEN_QUERY = 0x00000008;
internal const int TOKEN_ADJUST_PRIVILEGES = 0x00000020;
internal const string SE_SHUTDOWN_NAME = "SeShutdownPrivilege";
internal const int EWX_LOGOFF = 0x00000000;
internal const int EWX_SHUTDOWN = 0x00000001;
internal const int EWX_REBOOT = 0x00000002;
internal const int EWX_FORCE = 0x00000004;
internal const int EWX_POWEROFF = 0x00000008;
internal const int EWX_FORCEIFHUNG = 0x00000010;
private static void DoExitWin(int DoFlag)
...{
bool ok;
TokPriv1Luid tp;
IntPtr hproc = GetCurrentProcess();
IntPtr htok = IntPtr.Zero;
ok = OpenProcessToken(hproc, TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, ref htok);
tp.Count = 1;
tp.Luid = 0;
tp.Attr = SE_PRIVILEGE_ENABLED;
ok = LookupPrivilegeValue(null, SE_SHUTDOWN_NAME, ref tp.Luid);
ok = AdjustTokenPrivileges(htok, false, ref tp, 0, IntPtr.Zero, IntPtr.Zero);
ok = ExitWindowsEx(DoFlag, 0);
}
public static void Reboot()
...{
DoExitWin(EWX_FORCE | EWX_REBOOT);
}
public static void PowerOff()
...{
DoExitWin(EWX_FORCE | EWX_POWEROFF);
}
public static void LogOff()
...{
DoExitWin(EWX_FORCE | EWX_LOGOFF);
}
}
}