分享

获取系统串口号

 筱肆 2013-12-13


   

       
        分类:
            Windows C++ 编程
       

    2011-08-25 16:45
    1841人阅读
    评论(0)
    收藏
    举报
   



   


以下是我自己用的获取系统串口号的代码(VS2005)


INT_PTR CWifiComfWRTagDlg::GetSerialPort(CStringArray &arrCom) //遍历计算机中有哪些串口

{

 //CStringArray ary;

 arrCom.RemoveAll();

 HKEY   hkey;    

 LONG   lRes   =   RegOpenKeyEx(HKEY_LOCAL_MACHINE, _T( "HARDWARE\\DEVICEMAP\\SERIALCOMM"), NULL, KEY_QUERY_VALUE   |   KEY_ENUMERATE_SUB_KEYS   |   KEY_READ,   &hkey);    


 if   (lRes   ==   ERROR_SUCCESS)    

 {

  TCHAR   tchKey[MAX_PATH];    

  TCHAR   tchValue[20];

  DWORD   dwIndex   =   0;

  DWORD   dwType   =   REG_SZ;

  while(lRes   ==   ERROR_SUCCESS)

  {

   DWORD   dwCount   =   MAX_PATH;

   DWORD   dwVCount   =   20;

   lRes   =   RegEnumValue(hkey,   dwIndex++,   tchKey,   &dwCount,   NULL,   &dwType,   (LPBYTE)tchValue,   &dwVCount);


   if(lRes   ==   ERROR_SUCCESS)

   {

    if((dwVCount   >   0)   &&   (dwCount   >   0))

    {

     arrCom.Add(tchValue);

    }

   }

  }

 }

 RegCloseKey(hkey);




 return   arrCom.GetSize();

}


void CWifiComfWRTagDlg::ScanPort()

{

  CStringArray arry_str_COM;


 for(int i=0;i<GetSerialPort(arry_str_COM);i++)

 {

  ((CComboBox *)GetDlgItem(IDC_CMB_COM_NO))->InsertString(i,arry_str_COM.GetAt(i));

 }


 ((CComboBox *)GetDlgItem(IDC_CMB_COM_NO))->SetCurSel(0);

}


 


以下是转载


转载1


枚举串口四法


 


串口作为最基本的电脑通信 I/O 接口,其使用虽然在 PC 上越来越少,但是在工业仪器领域仍然用的相当普遍,由于笔者工作中需要用到串口,而且发现枚举串口至今仍未搞得很清楚,为此自己先整理下,希望大侠和同行们对我不懂和错误的地方指点一下。


 


1 、查询注册表


查询注册表的方法是网上见到的比较常见的方法,该方法就是使用编程方法读取注册表内信息,相当于用户通过在运行框内输入 ”regedit” (或 regedit32 )直接打开注册表,查看“ HKEY_LOCAL_MACHINE\HARDWARE\DEVICEMAP\SERIALCOMM ”项来获取串口信息。以下是源代码:


CString   strSerialList[256];  // 临时定义 256 个字符串组,因为系统最多也就 256 个


HKEY hKey;


LPCTSTR data_Set="HARDWARE\\DEVICEMAP\\SERIALCOMM\\";


long ret0 = (::RegOpenKeyEx(HKEY_LOCAL_MACHINE, data_Set, 0, KEY_READ, &hKey));


if(ret0 != ERROR_SUCCESS)


{


return -1;


}


int i = 0;


CHAR Name[25];


UCHAR szPortName[25];


LONG Status;


DWORD dwIndex = 0;


DWORD dwName;


DWORD dwSizeofPortName;


DWORD Type;


dwName = sizeof(Name);


dwSizeofPortName = sizeof(szPortName);


do


{


Status = RegEnumValue(hKey, dwIndex++, Name, &dwName, NULL, &Type,


      szPortName, &dwSizeofPortName);


if((Status == ERROR_SUCCESS)||(Status == ERROR_MORE_DATA))


{


strSerialList[i] = CString(szPortName);       // 串口字符串保存


i++;// 串口计数


   }


} while((Status == ERROR_SUCCESS)||(Status == ERROR_MORE_DATA));


RegCloseKey(hKey);


以上方法同样也可以实现对并口的查询,只要将 "HARDWARE \\ DEVICEMAP\\ SERIALCOMM\\" 用 "HARDWARE\\DEVICEMAP\\PARALLEL PORTS\\" 代替就行了。


比较:该方法时间最省,笔者在自己电脑上试过,在 1ms (少于 1ms 的我也不知道怎么编程计时)内即可完成;同时也可解决 usb 转串口设备的问题,比较实用,唯一缺点是,如果用户在装某些软硬件时在注册表中注册了虚拟串口之类的,用此法枚举得到的该类串口实际上是不能当串口用的。


 


2 、使用 EnumPort 方法


该方法调用 EnumPort () API 函数,该函数本身就是枚举电脑端口用的,它枚举的并非只有串口,所以必须对其所得串口进行分析选择,以下是源代码:


       int m_nSerialPortNum(0);// 串口计数


       CString          strSerialList[256];  // 临时定义 256 个字符串组


       LPBYTE pBite  = NULL;


       DWORD pcbNeeded = 0;  // bytes received or required


       DWORD pcReturned = 0;  // number of ports received


       m_nSerialPortNum = 0;


       // 获取端口信息,能得到端口信息的大小 pcbNeeded


       EnumPorts(NULL, 2, pBite, 0, &pcbNeeded, &pcReturned);


       pBite = new BYTE[pcbNeeded];


       // 枚举端口,能得到端口的具体信息 pBite 以及端口的的个数 pcReturned


       EnumPorts(NULL, 2, pBite, pcbNeeded, &pcbNeeded, &pcReturned);


       PORT_INFO_2 *pPort;


       pPort = (PORT_INFO_2*)pBite;


       for ( i = 0; i < pcReturned; i++)


       {


              CString str = pPort[i].pPortName;


              // 串口信息的具体确定


              if (str.Left(3) == "COM")


              {                  


                     strSerialList[m_nSerialPortNum] = str.Left(strlen(str) - 1);


                     //CString temp = str.Right(strlen(str) - 3);// 下面两行注释获取串口序号用


                     //m_nSerialPortNo[m_nSerialPortNum] = atoi(temp.Left(strlen(temp) - 1));


                     m_nSerialPortNum++;                


              }


       }


以上方法除了串口,还可以枚举所有的并口和打印机等接口,而且能找到虚拟串口(这些串口有些未使用时,在注册表和硬件设备管理器中是不能取得的)。但是该方法稍微耗时些,笔者在自己电脑上试过,大概需要几十 ms ,主要问题是该方法有些 usb 串口并不能查到,所以该方法并不可靠。


 


3 、依次打开串口的方法


该方法就是中规中矩的依次打开串口,看打开是否成功来判断串口的有无,该方法源代码如下:


       int m_nSerialPortNum(0);// 串口数


       CString          strSerialList[256];  // 临时定义 30 个字符串组


       int nCom = 0;


       int count = 0;


       HANDLE hCom;


       do {


              nCom++;


              strCom.Format("COM%d", nCom);


              hCom = CreateFile(strCom, 0, 0, 0,


                     OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0);


              if(INVALID_HANDLE_VALUE == hCom )


                     break;


              strSerialList[m_nSerialPortNum] = strCom;


              m_nSerialPortNum++;         


              CloseHandle(hCom);


       } while(1);


以上方法枚举的都是当前可用的串口,如果有一个串口当前被占用则其后的串口也将无法枚举得到,当然以上方法也可以改成调用 for 循环让其枚举打开 256 个串口的方法以避免上述情况,不过该方法比前两种更耗时(一般查找一个串口就要 15ms 左右),不过可以枚举得到所有当前可打开的串口,当然不能枚举得到一些虚拟串口。


 


4 、使用 SetupAPI 函数集的方法


此种方法是我所见过最简单的方法,之所以简单是因为已经有人将复杂的代码封装起来了,我只需像傻子一样调用就可以完成工作了,具体的说明请看
http://www./Cpp/W-P/system/hardwareinformation/article.php/c5721/
,下面给出本人调用该方法的例子代码:


       int m_nSerialPortNum(0);// 串口计数


       CString          strSerialList[256];  // 临时定义 256 个字符串组


       CArray<SSerInfo,SSerInfo&> asi;


       EnumSerialPorts(asi,TRUE);// 参数为 TRUE 时枚举当前可以打开的串口,

// 否则枚举所有串口


       m_nSerialPortNum = asi.GetSize();


       for (int i=0; i<asi.GetSize(); i++)


       {


              CString str = asi[i].strFrien dlyName;


       }


补充说明一下,使用该方法只要在你的程序中,添加“ EnumSerial.cpp ”和“ EnumSerial.h ”两个文件,并且将 Setupapi.lib 包含进你的工程文件中就行了,该方法时间上来说可能和第三种方法差不多,但该方法获取的串口完完全全就是硬件设备管理器中的串口。


以上是笔者对枚举串口几种方法的小结,有些没弄明白或含糊的地方,还请指正。


 


转载2


int CUtil::GetSystemPorts(CString *pNameList, CString *pPortList)  

{  

    CRegKey RegKey;  

    int nCount = 0;  

      

    if(RegKey.Open(HKEY_LOCAL_MACHINE, "Hardware\\DeviceMap\\SerialComm") == ERROR_SUCCESS)  


    {  

        while(true)  

        {  

            char ValueName[_MAX_PATH];  

            unsigned char ValueData[_MAX_PATH];  

            DWORD nValueSize = _MAX_PATH;  

            DWORD nDataSize = _MAX_PATH;  

            DWORD nType;  

              

            if(::RegEnumValue(HKEY(RegKey), nCount, ValueName, &nValueSize, NULL, &nType, ValueData, &nDataSize) == ERROR_NO_MORE_ITEMS)  


            {  

                break;  

            }  

              

            if(pNameList)  

                pNameList[nCount] = ValueName;  

              

            if(pPortList)  

                pPortList[nCount] = ValueData;  

              

            nCount++;  

        }  

    }  

      

    return nCount;  

}














更多
0






           
  • 上一篇:Windows Mobile 6 Professional SDK

  •            
  • 下一篇:VC获取当前程序文件的路径,文件名以及路径+文件名
    • 本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
      转藏 分享 献花(0

      0条评论

      发表

      请遵守用户 评论公约

      类似文章 更多