C# USB转串口突然拔出检测解决方案

go:http://blog.csdn.net/wuqiubin/article/details/2052182

 

最近做虚拟串口通讯程序(IRDA-USB-串口),由于是USB设备所以在通讯过程中有可能把串口拔出,程序需要实时检测到串口拔出。并把正在执行的任务结束关闭串口,给出提示。由于程序是C#做的,C#在操作底层比较不方便,需要调用较多的API函数。开始的方法是先捕获USB设备的拔出,再查找HKEY_LOCAL_MACHINE/HAEDWARE/DEVICEMAP/SERialCOM中的串口是否已经不存在。后来发现这种办法不可行.后来我想查找注册表的HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Control/DeviceClasses/a5dcbf10-6530-11d2-901f-00c04fb951ed} 中的相应项看是否LInked等于零,如果是就是你的usb设备拔出了。而且你还要检测是不是你的那个串口,这样算是可以解决问题.。之后我有想用异常的方法来解决这个问题,首先还是捕获USB设备的拔出消息,由于.NET并不知道串口的拔出,要是该串口正在使用,这是对串口的访问将会产生异常,所以通过捕获异常来处理是最好有方便的。代码如下:

        protected override void WndProc( ref Message m)
        {
           
if (m.Msg == 0x0219 )
            {
// 设备被拔出
                if (m.WParam.ToInt32() == 0x8004 ) // usb串口
                {
                   
if (对串口进行操作)
                    {
// 产生异常
                          关闭串口
                    }
                }
            }
           
base .WndProc( ref m);
        }

以下转正:http://www.cnblogs.com/AndyHai/archive/2007/07/25/830224.html

在.NET中探测U盘的插入/拔出

  有同学向我问这个问题,于是就Google了一下找到答案,不过是C下的,我将其改编成了C#的。

  当设备被插入/拔出的时候,WINDOWS会向每个窗体发送WM_DEVICECHANGE 消息,当消息的wParam 值等于 DBT_DEVICEARRIVAL 时,表示Media设备被插入并且已经可用;如果wParam值等于DBT_DEVICEREMOVECOMPLETE,表示Media设备已经被移出。

它们的lParam都指向一个 DEV_BROADCAST_HDR结构体,其原形如下:

1 typedef struct _DEV_BROADCAST_HDR
2 {
3     DWORD dbch_size;
4     DWORD dbch_devicetype;
5     DWORD dbch_reserved;
6 } DEV_BROADCAST_HDR, * PDEV_BROADCAST_HDR;

这个结构体仅仅是一个“头”(HDR),其后还有附加数据,dbch_size表示结构体实例的字节数,当其中的dbch_devicetype字段值等于 DBT_DEVTYP_VOLUME 时,表示当前设备是逻辑驱动器,且lParam实际上指向的应该是 DEV_BROADCAST_VOLUME 结构体实例(真佩服这种逻辑), DEV_BROADCAST_VOLUME 结构体原形如下:

1 typedef struct _DEV_BROADCAST_VOLUME {
2     DWORD dbcv_size;
3     DWORD dbcv_devicetype;
4     DWORD dbcv_reserved;
5     DWORD dbcv_unitmask;
6     WORD dbcv_flags;
7 } DEV_BROADCAST_VOLUME, * PDEV_BROADCAST_VOLUME;
其中 dbcv_unitmask 字段表示当前改变的驱动器掩码,第一位表示驱动器号A,第二位表示驱动器号B,第三位表示驱动器号C,以此类推…… dbcv_flags 表示驱动器的类别,如果等于1,则是光盘驱动器;如果是2,则是网络驱动器;如果是硬盘、U盘则都等于0

  所以,我只需要在程序中捕捉 WM_DEVICECHANGE 消息,然后根据具体情况去处理即可,下面是我的测试代码:

using System;
using System.Runtime.InteropServices;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;

namespace UDiskDetect
{
   
public partial class Form1 : Form
   
{
       
public Form1()
       
{
            InitializeComponent();
        }


       
private void Form1_Load(object sender, EventArgs e)
       
{

        }


        [StructLayout(LayoutKind.Sequential)]
       
struct DEV_BROADCAST_HDR
       
{
           
public UInt32 dbch_size;
           
public UInt32 dbch_devicetype;
           
public UInt32 dbch_reserved;
        }


        [StructLayout(LayoutKind.Sequential)]
       
struct DEV_BROADCAST_VOLUME
       
{
           
public UInt32 dbcv_size;
           
public UInt32 dbcv_devicetype;
           
public UInt32 dbcv_reserved;
           
public UInt32 dbcv_unitmask;
           
public UInt16 dbcv_flags;
        }


       
protected override void DefWndProc(ref Message m)
       
{
           
if (m.Msg == 0x0219)//WM_DEVICECHANGE
            {
               
switch (m.WParam.ToInt32())
               
{
                   
case 0x8000://DBT_DEVICEARRIVAL
                        {
                            DEV_BROADCAST_HDR dbhdr
= (DEV_BROADCAST_HDR)Marshal.PtrToStructure(m.LParam, typeof(DEV_BROADCAST_HDR));

                           
if (dbhdr.dbch_devicetype == 0x00000002)//DBT_DEVTYP_VOLUME
                            {
                                DEV_BROADCAST_VOLUME dbv
= (DEV_BROADCAST_VOLUME)Marshal.PtrToStructure(m.LParam, typeof(DEV_BROADCAST_VOLUME));
                               
if (dbv.dbcv_flags == 0)
                                    AddVolumes(GetVolumes(dbv.dbcv_unitmask));
                            }

                           
break;
                        }

                   
case 0x8004://DBT_DEVICEREMOVECOMPLETE
                        {
                            DEV_BROADCAST_HDR dbhdr
= (DEV_BROADCAST_HDR)Marshal.PtrToStructure(m.LParam, typeof(DEV_BROADCAST_HDR));

                           
if (dbhdr.dbch_devicetype == 0x00000002)//DBT_DEVTYP_VOLUME
                            {
                                DEV_BROADCAST_VOLUME dbv
= (DEV_BROADCAST_VOLUME)Marshal.PtrToStructure(m.LParam, typeof(DEV_BROADCAST_VOLUME));
                               
if (dbv.dbcv_flags == 0)
                                    RemoveVolumes(GetVolumes(dbv.dbcv_unitmask));
                            }

                           
break;
                        }

                }

            }

           
base.DefWndProc(ref m);
        }


       
/**//// <summary>
       
/// 根据驱动器掩码返回驱动器号数组
       
/// </summary>
       
/// <param name="Mask">掩码</param>
       
/// <returns>返回驱动器号数组</returns>

        public static char[] GetVolumes(UInt32 Mask)
       
{
            List
<char> Volumes = new List<char>();

           
for (int i = 0; i < 32; i++)
           
{
               
uint p = (uint)Math.Pow(2, i);
               
if ((p | Mask) == p)
               
{
                    Volumes.Add((
char)('A' + i));
                }

            }


           
return Volumes.ToArray();
        }


       
public void AddVolumes(char[] Volumes)
       
{
           
foreach (char volume in Volumes)
                listBox1.Items.Add(volume);
        }


       
public void RemoveVolumes(char[] Volumes)
       
{
           
foreach (char volume in Volumes)
                listBox1.Items.Remove(volume);
        }


    }

}

 

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值