@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 閱讀(9721)
評論(4) 編輯 收藏 引用 所屬分類:
MFC相關技術