我正在与可以随时插入或移除的USB转串口连接。我发现我可以使用WMI(特别是使用WMI Code Creator)来查询PC中的设备更改。
在下面生成的代码段中,订阅了Win32_DeviceChangeEvent。但是,此事件未显示哪个设备(例如USB,串行端口等)导致该事件。有没有办法只在插入或删除串行端口时接收通知?
为了澄清,代码点不来检测打开/关闭串口,它是检测是否新端口已添加到计算机或以前的端口已删除。
using System;
using System.Management;
using System.Windows.Forms;
namespace WMISample
{
public class WMIReceiveEvent
{
public WMIReceiveEvent()
{
try
{
WqlEventQuery query = new WqlEventQuery(
"SELECT * FROM Win32_DeviceChangeEvent");
ManagementEventWatcher watcher = new ManagementEventWatcher(query);
Console.WriteLine("Waiting for an event...");
watcher.EventArrived +=
new EventArrivedEventHandler(
HandleEvent);
// Start listening for events
watcher.Start();
// Do something while waiting for events
System.Threading.Thread.Sleep(10000);
// Stop listening for events
watcher.Stop();
return;
}
catch(ManagementException err)
{
MessageBox.Show("An error occurred while trying to receive an event: " + err.Message);
}
}
private void HandleEvent(object sender,
EventArrivedEventArgs e)
{
Console.WriteLine("Win32_DeviceChangeEvent event occurred.");
}
public static void Main()
{
WMIReceiveEvent receiveEvent = new WMIReceiveEvent();
return;
}
}
}
答案 0 :(得分:21)
我最终使用WMI和@Hans的建议来检查哪些串口是新的/缺失的。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Diagnostics.Contracts;
using System.IO.Ports;
using System.Management;
public static class SerialPortService
{
private static SerialPort _serialPort;
private static string[] _serialPorts;
private static ManagementEventWatcher arrival;
private static ManagementEventWatcher removal;
static SerialPortService()
{
_serialPorts = GetAvailableSerialPorts();
MonitorDeviceChanges();
}
/// <summary>
/// If this method isn't called, an InvalidComObjectException will be thrown (like below):
/// System.Runtime.InteropServices.InvalidComObjectException was unhandled
///Message=COM object that has been separated from its underlying RCW cannot be used.
///Source=mscorlib
///StackTrace:
/// at System.StubHelpers.StubHelpers.StubRegisterRCW(Object pThis, IntPtr pThread)
/// at System.Management.IWbemServices.CancelAsyncCall_(IWbemObjectSink pSink)
/// at System.Management.SinkForEventQuery.Cancel()
/// at System.Management.ManagementEventWatcher.Stop()
/// at System.Management.ManagementEventWatcher.Finalize()
///InnerException:
/// </summary>
public static void CleanUp()
{
arrival.Stop();
removal.Stop();
}
public static event EventHandler<PortsChangedArgs> PortsChanged;
private static void MonitorDeviceChanges()
{
try
{
var deviceArrivalQuery = new WqlEventQuery("SELECT * FROM Win32_DeviceChangeEvent WHERE EventType = 2");
var deviceRemovalQuery = new WqlEventQuery("SELECT * FROM Win32_DeviceChangeEvent WHERE EventType = 3");
arrival = new ManagementEventWatcher(deviceArrivalQuery);
removal = new ManagementEventWatcher(deviceRemovalQuery);
arrival.EventArrived += (o, args) => RaisePortsChangedIfNecessary(EventType.Insertion);
removal.EventArrived += (sender, eventArgs) => RaisePortsChangedIfNecessary(EventType.Removal);
// Start listening for events
arrival.Start();
removal.Start();
}
catch (ManagementException err)
{
}
}
private static void RaisePortsChangedIfNecessary(EventType eventType)
{
lock (_serialPorts)
{
var availableSerialPorts = GetAvailableSerialPorts();
if (!_serialPorts.SequenceEqual(availableSerialPorts))
{
_serialPorts = availableSerialPorts;
PortsChanged.Raise(null, new PortsChangedArgs(eventType, _serialPorts));
}
}
}
public static string[] GetAvailableSerialPorts()
{
return SerialPort.GetPortNames();
}
}
public enum EventType
{
Insertion,
Removal,
}
public class PortsChangedArgs : EventArgs
{
private readonly EventType _eventType;
private readonly string[] _serialPorts;
public PortsChangedArgs(EventType eventType, string[] serialPorts)
{
_eventType = eventType;
_serialPorts = serialPorts;
}
public string[] SerialPorts
{
get
{
return _serialPorts;
}
}
public EventType EventType
{
get
{
return _eventType;
}
}
}
MonitorDeviceChanges
方法实际上看到了所有设备更改(例如设备管理器),但检查串行端口只允许我们在事件发生变化时引发事件。
要使用该代码,只需订阅PortsChanged
事件,例如SerialPortService.PortsChanged += (sender1, changedArgs) => DoSomethingSerial(changedArgs.SerialPorts);
哦,.Raise
方法只是我在某个地方找到的扩展方法:
/// <summary>
/// Tell subscribers, if any, that this event has been raised.
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="handler">The generic event handler</param>
/// <param name="sender">this or null, usually</param>
/// <param name="args">Whatever you want sent</param>
public static void Raise<T>(this EventHandler<T> handler, object sender, T args) where T : EventArgs
{
// Copy to temp var to be thread-safe (taken from C# 3.0 Cookbook - don't know if it's true)
EventHandler<T> copy = handler;
if (copy != null)
{
copy(sender, args);
}
}
答案 1 :(得分:2)
没有。去了解SerialPort.GetPortNames()发生了什么。在窗口中收听WM_DEVICECHANGE消息可以为您提供更好的信息。
答案 2 :(得分:2)
注意:我试图将此作为对@ Pat的回答发表评论,但没有足够的声誉来做到这一点。
除了@ 2pietjuh2的评论之外,RaisePortsChangedIfNecessary()可以更改为以下内容:
private static void RaisePortsChangedIfNecessary(EventType eventType)
{
lock (_serialPorts)
{
var availableSerialPorts = GetAvailableSerialPorts();
if (eventType == EventType.Insertion)
{
var added = availableSerialPorts.Except(_serialPorts).ToArray();
_serialPorts = availableSerialPorts;
PortsChanged.Raise(null, new PortsChangedArgs(eventType, added));
}
else if (eventType == EventType.Removal)
{
var removed = _serialPorts.Except(availableSerialPorts).ToArray();
_serialPorts = availableSerialPorts;
PortsChanged.Raise(null, new PortsChangedArgs(eventType, removed));
}
}
}
引发的事件包括插入/删除的串行端口,而不是插入/移除后可用的串行端口列表。
答案 3 :(得分:1)
这是我前段时间写过的DeviceChangeEvents
通知类的精简版本,但我从未完全完成它。除了PortArrived事件之外我删除了所有内容,因为它非常难看。
using System;
using System.ComponentModel;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Windows.Forms;
public sealed class PortArrivalEventArgs : EventArgs
{
public string Name { get; private set; }
public PortArrivalEventArgs(string name) { Name = name; }
}
public static class DeviceChangeEvents
{
#region Events
#region PortArrived
private static object PortArrivedEvent = new Object();
public static event EventHandler<PortArrivalEventArgs> PortArrived
{
add { AddEvent(PortArrivedEvent, value); }
remove { RemoveEvent(PortArrivedEvent, value); }
}
private static void FirePortArrived(IntPtr lParam)
{
EventHandler<PortArrivalEventArgs> handler
= (EventHandler<PortArrivalEventArgs>)events[PortArrivedEvent];
if (handler != null)
{
string portName = Marshal.PtrToStringAuto((IntPtr)((long)lParam + 12));
handler(null, new PortArrivalEventArgs(portName));
}
}
#endregion
#endregion
#region Internal
private static EventHandlerList events = new EventHandlerList();
private static MessageWindow messageWindow = null;
private static void AddEvent(object key, Delegate value)
{
events.AddHandler(key, value);
if (messageWindow == null)
messageWindow = new MessageWindow();
}
private static void RemoveEvent(object key, Delegate value)
{
events.RemoveHandler(key, value);
// In the more complete version of DeviceChangedEvents, System.ComponentModel.EventHandlerList
// is replaced by an identical event storage object which exposes a count of the number of
// handlers installed. It also removes empty handler stubs. Both of these are required
// to safely destroy the message window when the last handler is removed.
//if (messageWindow != null && events.Count == 0)
// messageWindow.DestroyHandle();
}
#endregion
private sealed class MessageWindow : NativeWindow
{
public MessageWindow()
{
CreateParams cp = new CreateParams();
cp.Caption = GetType().FullName;
// NOTE that you cannot use a "message window" for this broadcast message
//if (Environment.OSVersion.Platform == PlatformID.Win32NT)
// cp.Parent = (IntPtr)(-3); // HWND_MESSAGE
//Debug.WriteLine("Creating MessageWindow " + cp.Caption);
CreateHandle(cp);
}
const int WM_DESTROY = 0x02;
const int WM_DEVICECHANGE = 0x219;
enum DBT
{
DEVICEARRIVAL = 0x8000,
}
protected override void WndProc(ref Message m)
{
if (m.Msg == WM_DESTROY)
{
messageWindow = null;
}
else if (m.Msg == WM_DEVICECHANGE)
{
DBT changeType = (DBT)m.WParam;
int deviceType = m.LParam == IntPtr.Zero ? 0 : Marshal.ReadInt32(m.LParam, 4);
Debug.WriteLine(String.Format("WM_DEVICECHANGE changeType = {0}, deviceType = {1}", changeType, deviceType));
switch (changeType)
{
case DBT.DEVICEARRIVAL:
switch (deviceType)
{
case 3: // DBT_DEVTYP_PORT
FirePortArrived(m.LParam);
break;
}
break;
}
}
base.WndProc(ref m);
}
}
}
答案 4 :(得分:0)
您的设备更改事件可与WMI - PNP实体一起使用。以下内容将返回设备详细信息 - 在下面的代码中显示设备名称。
Dim moReturn As Management.ManagementObjectCollection
Dim moSearch As Management.ManagementObjectSearcher
Dim mo As Management.ManagementObject
moSearch = New Management.ManagementObjectSearcher("Select * from Win32_PnPEntity")
moReturn = moSearch.Get
For Each mo In moReturn
If CStr(mo.Properties.Item("Name").Value).Contains("Prolific") Then
returns something like: "Prolific USB-to-Serial Comm Port (COM17)"
txtStatus.Text &= CStr(mo.Properties.Item("Name").Value) & vbCrLf
End If
Next
另请参阅代码以访问可用于过滤或监控更改的其他PNP属性:
On Error Resume Next
strComputer = "."
Set objWMIService = GetObject("winmgmts:\\" & strComputer & "\root\cimv2")
Set colItems = objWMIService.ExecQuery("Select * from Win32_PnPEntity",,48)
For Each objItem in colItems
"Availability: " & objItem.Availability
"Caption: " & objItem.Caption
"ClassGuid: " & objItem.ClassGuid
"ConfigManagerErrorCode: " & objItem.ConfigManagerErrorCode
"ConfigManagerUserConfig: " & objItem.ConfigManagerUserConfig
"CreationClassName: " & objItem.CreationClassName
"Description: " & objItem.Description
"DeviceID: " & objItem.DeviceID
"ErrorCleared: " & objItem.ErrorCleared
"ErrorDescription: " & objItem.ErrorDescription
"InstallDate: " & objItem.InstallDate
"LastErrorCode: " & objItem.LastErrorCode
"Manufacturer: " & objItem.Manufacturer
"Name: " & objItem.Name
"PNPDeviceID: " & objItem.PNPDeviceID
"PowerManagementCapabilities: " & objItem.PowerManagementCapabilities
"PowerManagementSupported: " & objItem.PowerManagementSupported
"Service: " & objItem.Service
"Status: " & objItem.Status
"StatusInfo: " & objItem.StatusInfo
"SystemCreationClassName: " & objItem.SystemCreationClassName
"SystemName: " & objItem.SystemName
Next