以编程方式禁用/启用网络接口
Posted
技术标签:
【中文标题】以编程方式禁用/启用网络接口【英文标题】:Programmatically disable/enable network interface 【发布时间】:2010-10-26 00:44:00 【问题描述】:我正在尝试提出一种以编程方式启用/禁用网卡的解决方案 - 我已经进行了大量研究,但在 XP 和 Vista 环境中似乎没有一个可行的解决方案。我在说的是,如果您进入控制面板“网络连接”,右键单击一个并选择启用或禁用。理想情况下,我想使用一个库,但如果情况变得更糟,我想我可以调用一个命令行应用程序,但这绝对是最坏的情况。以下是我迄今为止尝试过的方法以及失败的地方/原因:
上一篇:
How to programmatically enable/disable network interfaces? (Windows XP)
列出了几个方法 - 第一个是使用 netsh,这似乎与使用 IPHelper 函数 SetIfEntry() 相同。问题在于它将接口设置为管理启用或禁用,而不是正常启用/禁用,因此它实际上并没有关闭 NIC。
提出的另一个解决方案是使用 WMI,特别是 Win32_NetworkAdapter 类,它具有启用和禁用方法:
http://msdn.microsoft.com/en-us/library/aa394216(VS.85).aspx
不错吧?在 Vista 中工作正常,这些方法在普通 XP 安装中不存在...
另一个建议是使用真正使用 SetupAPI 的 DevCon,特别是带有 DICS_ENABLE 的 SetupDiSetClassInstallParams()。在这个精彩的课程上花费了无数小时,并尝试在全局级别以及特定配置级别(以及每个组合)禁用/启用设备后,它也不能始终如一地工作 - 有时工作正常,但其他时候在设备管理器中禁用设备,但仍将其留在网络连接中并运行。
然后我尝试使用 INetConnection 接口,特别是 INetConnection->Connect/Disconnect:
http://msdn.microsoft.com/en-us/library/aa365084(VS.85).aspx
但我从来没有让这对我的 Vista 或 XP 测试盒上的连接产生任何影响。
最后,我找到了这个名为 ToggleNic 的 C# 脚本:
http://channel9.msdn.com/playground/Sandbox/154712/
看起来它以某种方式通过 Shell 以有效地导致右键单击行为。限制(至少是这个实现)是它在非英语系统上不起作用(未经修改),我需要我的系统来使用。公平地说,这个解决方案看起来是最可行的,但我对 C# 的熟悉程度很低,我找不到它使用的 API 是否在 C++ 中可用。
任何帮助或见解将不胜感激 - 或有关如何完成切换脚本在 C++ 中的作用的想法。谢谢!
【问题讨论】:
【参考方案1】:ManagementClass managementClass = new ManagementClass("Win32_NetworkAdapter");
ManagementObjectCollection mgmtObjectColl = managementClass.GetInstances();
ManagementObject myObject = null;
foreach (ManagementObject mgmtObject in mgmtObjectColl)
if (mgmtObject["NetConnectionID"] != null && mgmtObject["NetConnectionID"].Equals("Local Area Connection"))
Console.WriteLine("found");
myObject = mgmtObject;
object result = mgmtObject.InvokeMethod("Disable", null);
//Console.WriteLine("0, 1", mgmtObject["Name"], mgmtObject["NetConnectionID"]);
object result3 = myObject.InvokeMethod("Enable", null);
【讨论】:
【参考方案2】:'==========================================================================
' NAME: 1unconnectedNIC.vbs
'
' COMMENT: This file will disable and rename the unconnected NIC on the
' local server.
'
' Usage:
' cscript 1unconnectedNIC.vbs
' Alex John
'==========================================================================
strComputer = "."
Set objShell = CreateObject("WScript.Shell")
Set objWMIService = GetObject("winmgmts:impersonationLevel=impersonate!\\" & strComputer & "\root\cimv2")
Set colAdapters = objWMIService.ExecQuery("SELECT * FROM Win32_NetworkAdapter where NetConnectionStatus = '7'" )
'For every adapter that is disconnected (has a red X by it)
For Each objAdapter in colAdapters
Wscript.Echo "Net Connection ID: " & objAdapter.NetConnectionID
Wscript.Echo "....Disabling NIC"
'Disable the NIC
objShell.Run "netsh interface set interface name=" & chr(34) & objAdapter.NetConnectionID & chr(34) & " admin=disabled", 0, true
Wscript.Echo "....NIC Disabled"
WScript.Sleep 10000
'Rename the NIC
Wscript.Echo "....Renaming NIC to 'Not Used'"
objShell.Run "netsh interface set interface name=" & chr(34) & objAdapter.NetConnectionID & chr(34) & " newname=" & chr(34) & "Not Used" & n & chr(34), 0, true
Next
Wscript.Echo "Finished"
【讨论】:
【参考方案3】:找到这个基本上是右键自动化的解决方案:http://www.wilderssecurity.com/showthread.php?t=265836 适用于 Windows XP 和 Windows 7
还有一个是 Novell 写的,它是 DevCon 之上的一个包装器: http://www.novell.com/communities/node/2338/network-configuration-command-line-control
【讨论】:
【参考方案4】:经过大量搜索后,我开始深入研究 DEVCON。在意识到这些方法必须是可调用的之后,我做了一些搜索,发现了这个: http://www.codeproject.com/KB/cs/HardwareHelper.aspx
它至少帮助解决了我从 Open*** 启用和禁用 TAP 适配器的问题。
【讨论】:
【参考方案5】:在更多平台和更多方法上进行测试后,我基本上已经放弃了这个功能(至少出于我的目的)。对我来说,问题是我想要在 90% 以上的情况下工作,而现实情况是,我能想到的一切都接近 70%。具有讽刺意味的是,它实际上与普通的 Windows 方法一样不稳定。对于那些仍然想走这条危险道路的人,这是我发现的:
在上述 API 直接方法中,最一致的方法是使用 SetupAPI (SetupDiSetClassInstallParams) - 我遇到的最大问题是有时它会进入需要重新启动的状态,并且在那之前,任何改变都不会奏效。使用它时唯一需要注意的另一件事是设备有两个配置文件,因此在某些情况下您需要同时切换它们。 DDK 包含 devcon 工具的源代码,它准确地向您展示了如何做所有事情。这最终看起来像是最接近右键单击,但它仍然表现出一些网络连接没有的奇怪行为。这种方法似乎在大约 70% 的时间内都有效(在测试和测试系统中)。
从整体破解方法中,我发现最好的方法不是使用 ToggleNIC 所做的技术,而是使用 IShellFolder 的东西——这允许您使用与语言无关的 GetCommandString。问题是在 XP 下 GetCommandString 没有返回任何东西(哦,高兴),但似乎“启用”和“禁用”的菜单 ID 是一致的(分别为 16 和 17),所以如果我未能 GetCommandString ,我只是回到菜单 ID 上。要切换,只需调用 InvokeCommand ,如果它返回一个字符串,或者如果它没有返回菜单 ID。这样做的问题是,就像正常的 Windows 方式一样,有时它不起作用,也没有给你任何关于正在发生的事情或失败原因的指示。这种方法似乎在大约 70% 的时间里也有效,但更难判断是否出现问题,而且会弹出正常的“正在启用界面...”文本。
希望这对其他人有所帮助 - 如果有人设法找到另一种适用于更多情况的方法,我很想听听!
【讨论】:
我发现 ToggleNIC 解决方案非常可靠,但有趣的是,我有时只看到“启用界面..”文本,这非常令人沮丧。【参考方案6】:ToggleNic 通过 COM 使用 Shell32。 在 c 中,这由标准库“SHLOBJ.H”“SHELLAPI.H”和 IShellDispatch 类使用。
【讨论】:
【参考方案7】:尝试“Shell 网络接口”C++ 代码here。这应该在 XP 和更高版本下工作。
【讨论】:
这就是我上面提到的 INetConnection 接口——那个链接说它是 Shell API,但它实际上是 Internet 连接防火墙的一部分……正如我上面的链接所提到的,它有两个问题——首先是,它不起作用(似乎没有启用以禁用任何东西),其次是它已被没有类似功能的 Windows 防火墙 API 取代。以上是关于以编程方式禁用/启用网络接口的主要内容,如果未能解决你的问题,请参考以下文章
如何以编程方式检测是不是在 Windows 桌面应用程序中启用/禁用了 javascript? (网络浏览器控件)
Android:如何以编程方式启用/禁用 Wifi 或 Internet 连接