• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            franksunny的個人技術空間
            獲得人生中的成功需要的專注與堅持不懈多過天才與機會。 ——C.W. Wendte
            @import url(http://m.shnenglu.com/CuteSoft_Client/CuteEditor/Load.ashx?type=style&file=SyntaxHighlighter.css);@import url(/css/cuteeditor.css);

             

            枚舉串口四法

             

            串口作為最基本的電腦通信 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.codeguru.com/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 包含進你的工程文件中就行了,該方法時間上來說可能和第三種方法差不多,但該方法獲取的串口完完全全就是硬件設備管理器中的串口。

            以上是筆者對枚舉串口幾種方法的小結,有些沒弄明白或含糊的地方,還請指正。

             

             

            posted on 2006-10-10 23:31 frank.sunny 閱讀(9749) 評論(4)  編輯 收藏 引用 所屬分類: MFC相關技術

            FeedBack:
            # re: 枚舉串口四法
            2006-10-11 12:13 | 開個玩笑
            好,頂一下  回復  更多評論
              
            # re: 枚舉串口四法
            2007-09-13 18:43 | 1111
            但是得不到com口對應的設備名字   回復  更多評論
              
            # re: 枚舉串口四法
            2007-09-14 09:28 | freanksunny
            @1111
            你想得到具體怎么樣的串口設備名字?
              回復  更多評論
              
            # re: 枚舉串口四法
            2011-07-05 14:39 | No
            第一個方法代碼有問題,,
            需要把這兩句加到do while循環的中
            dwName = sizeof(Name);

            dwSizeofPortName = sizeof(szPortName);

            每次調用RegEnumValue 都需要重設這2個值。
            否則只能重復得到dwIndex==0 的那個值。  回復  更多評論
              

            常用鏈接

            留言簿(13)

            隨筆分類

            個人其它博客

            基礎知識鏈接

            最新評論

            閱讀排行榜

            評論排行榜

            一本一道久久综合狠狠老| 精品久久久久久无码中文字幕一区| 77777亚洲午夜久久多喷| 99久久99久久精品国产| 区亚洲欧美一级久久精品亚洲精品成人网久久久久 | 久久综合成人网| 久久久久人妻一区精品色| 精品国产乱码久久久久久浪潮 | 欧美无乱码久久久免费午夜一区二区三区中文字幕 | 久久精品国产免费观看| 97久久精品人人做人人爽| 久久无码高潮喷水| 国产精品青草久久久久婷婷| 99精品国产免费久久久久久下载| 国产91久久精品一区二区| 久久久久久精品无码人妻| 久久精品国产72国产精福利| 国产精品久久网| 国产成人精品白浆久久69| 久久精品国产99久久久古代| 久久综合久久性久99毛片| 国产成人久久久精品二区三区| 中文字幕无码免费久久| 国产精品99久久久久久宅男小说| 欧美伊香蕉久久综合类网站| 久久久久无码精品国产不卡| 亚洲国产另类久久久精品小说| 亚洲欧美一级久久精品| 日本精品久久久久影院日本| 久久精品成人免费国产片小草| 99久久久久| 久久精品国产亚洲Aⅴ蜜臀色欲 | 中文字幕日本人妻久久久免费| 久久99这里只有精品国产| 日本WV一本一道久久香蕉| 亚洲国产成人久久精品99| 一本色道久久88综合日韩精品 | 久久婷婷五月综合色99啪ak| 久久国产精品免费一区| 久久亚洲精品国产亚洲老地址 | 精品久久久久久无码人妻蜜桃|