VC枚举串口端口应用

本文实例讲述了VC串口端口应用,分享给大家供大家参考。具体用法分析如下:

串口作为最基本的电脑通信 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 包含进你的工程文件中就行了,该方法时间上来说可能和第三种
方法差不多,但该方法获取的串口完完全全就是硬件设备管理器中的串口。

希望本文所述对大家的VC程序设计有所帮助。

(0)

相关推荐

  • VC实现的病毒专杀工具完整实例

    本文实例讲述了VC实现的病毒专杀工具的方法.非常实用,分享给大家供大家参考.具体实现方法如下: 如今病毒木马蠕虫层出不穷,变种也是一个接一个.反病毒公司以及各大安全公司随着影响很大的病毒的出现都会免费提供病毒专杀工具,这个举措对普通用户来说确实很有帮助.其实写病毒专杀工具也不像大家想象的那么神秘,利用SDK写个控制台程序来实现病毒专杀,因无须写图形界面,所以简便快捷!你自己也能写!不信?就接着看吧^_^ 废话不说了,接下来就开始谈谈病毒专杀工具的思路及实现方法. 本文中讲解的病毒专杀工具是针对木

  • VC++获得当前进程运行目录的方法

    本文实例讲述了VC++获得当前进程运行目录的方法,分享给大家供大家参考.具体实现方法如下: 该实例主要通过利用windows API获得,具体代码如下: 复制代码 代码如下: CString GetAppPath() {   CString strAppPath;  // 保存结果   TCHAR szModuleFileName[MAX_PATH]; // 全路径名   TCHAR drive[_MAX_DRIVE];  // 盘符名称,比如说C盘啊,D盘啊   TCHAR dir[_MAX_

  • VC++中进程与多进程管理的方法详解

    本文实例讲述了VC++中进程与多进程管理的方法,分享给大家供大家参考.具体方法分析如下: 摘要: 本文主要介绍了多任务管理中的多进程管理技术,对进程的互斥运行.子进程的创建与结束等作了较详细的阐述. 关键词: VC++6.0:进程:环境变量:子进程 进程 进程是当前操作系统下一个被加载到内存的.正在运行的应用程序的实例.每一个进程都是由内核对象和地址空间所组成的,内核对象可以让系统在其内存放有关进程的统计信息并使系统能够以此来管理进程,而地址空间则包括了所有程序模块的代码和数据以及线程堆栈.堆分

  • VC判断一个文件为目录的方法

    本文实例讲述了VC判断一个文件为目录的方法,分享给大家供大家参考.具体实现方法如下: 这是一个自定义函数,用于判断一个文件是否为目录: 复制代码 代码如下: /**  * check whether a file is a directory  @return true if is a directory, else false(if file not exists, false)  */ __declspec(dllexport) bool IsDirectory(const char* fi

  • VC判断进程是否具有administrator权限的方法

    本文实例讲述了VC判断进程是否具有admin权限的方法.是的话返回TRUE,否则为FALSE.分享给大家供大家参考.具体实现方法如下: 复制代码 代码如下: static BOOL IsAdmin(void)  {       HANDLE                   hAccessToken;       BYTE                     InfoBuffer[1024];          PTOKEN_GROUPS            ptgGroups;    

  • VC定时器的用法实例详解

    本文实例讲述了VC中定时器的用法,分享给大家供大家参考.具体用法分析如下: 定时器在VC中的使用频繁,其原型为: 复制代码 代码如下: WINUSERAPI UINT WINAPI SetTimer ( HWND hWnd , UINT nIDEvent, UINT uElapse, TIMERPROC lpTimerFunc); 其中的参数用法如下: hWnd       是欲设置定时器的窗体句柄.定时时间到时,系统会向该窗体发送WM_TIMER消息. nIDEvent     定时器标识符.

  • VC文件目录常见操作实例汇总

    一般来说,在VC里文件操作有很多,本文在这里收录了一些常见的函数,分享给大家供大家参考.具体如下: 1. 判断一个目录是否存在 复制代码 代码如下: #include "windows.h" //参数: strPath: 目录的完整路径,注意不要以'/'结尾 //返回值: 如果为目录,返回真,否则返回假 BOOL FolderExist(CString strPath)  {     WIN32_FIND_DATA wfd;     BOOL rValue = FALSE;     H

  • 判断指定的进程或程序是否存在方法小结(vc等)

    一.判断指定程序名的进程是否存在     BOOL EnumWindows( WNDENUMPROC lpEnumFunc, // pointer to callback function LPARAM lParam //   application-defined value);        The EnumWindows function enumerates all top-level windows on the screen by passing the handle to each

  • VC程序在Win32环境下动态链接库(DLL)编程原理

    本文详细讲述了VC程序在Win32环境下动态链接库(DLL)编程原理.分享给大家供大家参考.具体分析如下: 一般比较大的应用程序都由很多模块组成,这些模块分别完成相对独立的功能,它们彼此协作来完成整个软件系统的工作.其中可能存在一些模块的功能较为通用,在构造其它软件系统时仍会被使用.在构造软件系统时,如果将所有模块的源代码都静态编译到整个应用程序EXE文件中,会产生一些问题:一个缺点是增加了应用程序的大小,它会占用更多的磁盘空间,程序运行时也会消耗较大的内存空间,造成系统资源的浪费:另一个缺点是

  • VC多线程编程详解

    本文实例讲述了VC多线程编程概念与技巧,分享给大家供大家参考.具体分析如下: 一.多线程编程要点 线程是进程的一条执行路径,它包含独立的堆栈和CPU寄存器状态,每个线程共享所有的进程资源,包括打开的文件.信号标识及动态分配的内存等.一个进程内的所有线程使用同一个地址空间,而这些线程的执行由系统调度程序控制,调度程序决定哪个线程可执行以及什么时候执行线程.线程有优先级别,优先权较低的线程必须等到优先权较高的线程执行完后再执行.在多处理器的机器上,调度程序可将多个线程放到不同的处理器上去运行,这样可

随机推荐