枚举串口

 

  1. 串口作为最基本的电脑通信 I/O 接口,其使用虽然在 PC 上越来越少,但是在工业仪器领域仍然用的相当普遍,由于笔者工作中需要用到串口,而且发现枚举串口至今仍未搞得很清楚,为此自己先整理下,希望大侠和同行们对我不懂和错误的地方指点一下。 
  2.   
  3. 1 、查询注册表 
  4. 查询注册表的方法是网上见到的比较常见的方法,该方法就是使用编程方法读取注册表内信息,相当于用户通过在运行框内输入 ”regedit” (或 regedit32 )直接打开注册表,查看“ HKEY_LOCAL_MACHINE/HARDWARE/DEVICEMAP/SERIALCOMM ”项来获取串口信息。以下是源代码: 
  5. CString   strSerialList[256];  // 临时定义 256 个字符串组,因为系统最多也就 256 个 
  6. HKEY hKey; 
  7. LPCTSTR data_Set="HARDWARE//DEVICEMAP//SERIALCOMM//"
  8. long ret0 = (::RegOpenKeyEx(HKEY_LOCAL_MACHINE, data_Set, 0, KEY_READ, &hKey)); 
  9. if(ret0 != ERROR_SUCCESS) 
  10. return -1; 
  11. int i = 0; 
  12. CHAR Name[25]; 
  13. UCHAR szPortName[25]; 
  14. LONG Status; 
  15. DWORD dwIndex = 0; 
  16. DWORD dwName; 
  17. DWORD dwSizeofPortName; 
  18. DWORD Type; 
  19. dwName = sizeof(Name); 
  20. dwSizeofPortName = sizeof(szPortName); 
  21. do 
  22. Status = RegEnumValue(hKey, dwIndex++, Name, &dwName, NULL, &Type, 
  23.       szPortName, &dwSizeofPortName); 
  24. if((Status == ERROR_SUCCESS)||(Status == ERROR_MORE_DATA)) 
  25. strSerialList[i] = CString(szPortName);       // 串口字符串保存 
  26. i++;// 串口计数 
  27.    } 
  28. while((Status == ERROR_SUCCESS)||(Status == ERROR_MORE_DATA)); 
  29. RegCloseKey(hKey); 
  30. 以上方法同样也可以实现对并口的查询,只要将 "HARDWARE // DEVICEMAP// SERIALCOMM//" 用 "HARDWARE//DEVICEMAP//PARALLEL PORTS//" 代替就行了。 
  31. 比较:该方法时间最省,笔者在自己电脑上试过,在 1ms (少于 1ms 的我也不知道怎么编程计时)内即可完成;同时也可解决 usb 转串口设备的问题,比较实用,唯一缺点是,如果用户在装某些软硬件时在注册表中注册了虚拟串口之类的,用此法枚举得到的该类串口实际上是不能当串口用的。 
  32.   
  33. 2 、使用 EnumPort 方法 
  34. 该方法调用 EnumPort () API 函数,该函数本身就是枚举电脑端口用的,它枚举的并非只有串口,所以必须对其所得串口进行分析选择,以下是源代码: 
  35.        int m_nSerialPortNum(0);// 串口计数 
  36.        CString          strSerialList[256];  // 临时定义 256 个字符串组 
  37.        LPBYTE pBite  = NULL; 
  38.        DWORD pcbNeeded = 0;  // bytes received or required 
  39.        DWORD pcReturned = 0;  // number of ports received 
  40.        m_nSerialPortNum = 0; 
  41.        // 获取端口信息,能得到端口信息的大小 pcbNeeded 
  42.        EnumPorts(NULL, 2, pBite, 0, &pcbNeeded, &pcReturned); 
  43.        pBite = new BYTE[pcbNeeded]; 
  44.        // 枚举端口,能得到端口的具体信息 pBite 以及端口的的个数 pcReturned 
  45.        EnumPorts(NULL, 2, pBite, pcbNeeded, &pcbNeeded, &pcReturned); 
  46.        PORT_INFO_2 *pPort; 
  47.        pPort = (PORT_INFO_2*)pBite; 
  48.        for ( i = 0; i < pcReturned; i++) 
  49.        { 
  50.               CString str = pPort[i].pPortName; 
  51.               // 串口信息的具体确定 
  52.               if (str.Left(3) == "COM"
  53.               {                   
  54.                      strSerialList[m_nSerialPortNum] = str.Left(strlen(str) - 1); 
  55.                      //CString temp = str.Right(strlen(str) - 3);// 下面两行注释获取串口序号用 
  56.                      //m_nSerialPortNo[m_nSerialPortNum] = atoi(temp.Left(strlen(temp) - 1)); 
  57.                      m_nSerialPortNum++;                 
  58.               } 
  59.        } 
  60. 以上方法除了串口,还可以枚举所有的并口和打印机等接口,而且能找到虚拟串口(这些串口有些未使用时,在注册表和硬件设备管理器中是不能取得的)。但是该方法稍微耗时些,笔者在自己电脑上试过,大概需要几十 ms ,主要问题是该方法有些 usb 串口并不能查到,所以该方法并不可靠。 
  61.   
  62. 3 、依次打开串口的方法 
  63. 该方法就是中规中矩的依次打开串口,看打开是否成功来判断串口的有无,该方法源代码如下: 
  64.        int m_nSerialPortNum(0);// 串口数 
  65.        CString          strSerialList[256];  // 临时定义 30 个字符串组 
  66.        int nCom = 0; 
  67.        int count = 0; 
  68.        HANDLE hCom; 
  69.        do { 
  70.               nCom++; 
  71.               strCom.Format("COM%d", nCom); 
  72.               hCom = CreateFile(strCom, 0, 0, 0, 
  73.                      OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); 
  74.               if(INVALID_HANDLE_VALUE == hCom ) 
  75.                      break
  76.               strSerialList[m_nSerialPortNum] = strCom; 
  77.               m_nSerialPortNum++;          
  78.               CloseHandle(hCom); 
  79.        } while(1); 
  80. 以上方法枚举的都是当前可用的串口,如果有一个串口当前被占用则其后的串口也将无法枚举得到,当然以上方法也可以改成调用 for 循环让其枚举打开 256 个串口的方法以避免上述情况,不过该方法比前两种更耗时(一般查找一个串口就要 15ms 左右),不过可以枚举得到所有当前可打开的串口,当然不能枚举得到一些虚拟串口。 
  81.   
  82. 4 、使用 SetupAPI 函数集的方法 
  83. 此种方法是我所见过最简单的方法,之所以简单是因为已经有人将复杂的代码封装起来了,我只需像傻子一样调用就可以完成工作了,具体的说明请看 http://www.codeguru.com/Cpp/W-P/system/hardwareinformation/article.php/c5721/ ,下面给出本人调用该方法的例子代码: 
  84.        int m_nSerialPortNum(0);// 串口计数 
  85.        CString          strSerialList[256];  // 临时定义 256 个字符串组 
  86.        CArray<SSerInfo,SSerInfo&> asi; 
  87.        EnumSerialPorts(asi,TRUE);// 参数为 TRUE 时枚举当前可以打开的串口, 
  88. // 否则枚举所有串口 
  89.        m_nSerialPortNum = asi.GetSize(); 
  90.        for (int i=0; i<asi.GetSize(); i++) 
  91.        { 
  92.               CString str = asi[i].strFrien dlyName; 
  93.        } 
  94. 补充说明一下,使用该方法只要在你的程序中,添加“ EnumSerial.cpp ”和“ EnumSerial.h ”两个文件,并且将 Setupapi.lib 包含进你的工程文件中就行了,该方法时间上来说可能和第三种方法差不多,但该方法获取的串口完完全全就是硬件设备管理器中的串口
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值