VC串口编程的实例2

矫情吗;* 2022-08-10 05:00 279阅读 0赞

  本例程采用异步串口操作,我们只介绍软件部分,RS485接口接线方法不作介绍,感兴趣的读者可以查阅相关资料。
  打开VC++6.0,新建基于对话框的工程RS485Comm,在主对话框窗口IDD_RS485COMM_DIALOG上添加两个按钮,ID分别为IDC_SEND和IDC_RECEIVE,标题分别为“发送”和“接收”;添加一个静态文本框IDC_DISP,用于显示串口接收到的内容。

1. 添加串口的全局变量

在RS485CommDlg.cpp文件中添加全局变量:

  1. HANDLE hCom; //全局变量

2. 初始化串口参数

  串口句柄在RS485CommDlg.cpp文件中的OnInitDialog()函数添加如下代码:

  1. hCom=CreateFile("COM1",//COM1口
  2. GENERIC_READ|GENERIC_WRITE, //允许读和写
  3. 0, //独占方式
  4. NULL,
  5. OPEN_EXISTING, //打开而不是创建
  6. FILE_ATTRIBUTE_NORMAL|FILE_FLAG_OVERLAPPED, //重叠方式
  7. NULL);
  8. if(hCom==(HANDLE)-1)
  9. {
  10. AfxMessageBox("打开COM失败!");
  11. return FALSE;
  12. }
  13. SetupComm(hCom,100,100); //输入缓冲区和输出缓冲区的大小都是100
  14. COMMTIMEOUTS TimeOuts;
  15. //设定读超时
  16. TimeOuts.ReadIntervalTimeout=MAXDWORD;
  17. TimeOuts.ReadTotalTimeoutMultiplier=0;
  18. TimeOuts.ReadTotalTimeoutConstant=0;
  19. //在读一次输入缓冲区的内容后读操作就立即返回,
  20. //而不管是否读入了要求的字符。
  21. //设定写超时
  22. TimeOuts.WriteTotalTimeoutMultiplier=100;
  23. TimeOuts.WriteTotalTimeoutConstant=500;
  24. SetCommTimeouts(hCom,&TimeOuts); //设置超时
  25. DCB dcb;
  26. GetCommState(hCom,&dcb);
  27. dcb.BaudRate=9600; //波特率为9600
  28. dcb.ByteSize=8; //每个字节有8位
  29. dcb.Parity=NOPARITY; //无奇偶校验位
  30. dcb.StopBits=TWOSTOPBITS; //两个停止位
  31. SetCommState(hCom,&dcb);
  32. PurgeComm(hCom,PURGE_TXCLEAR|PURGE_RXCLEAR);

3.添加发送和接收函数

  分别双击IDC_SEND按钮和IDC_RECEIVE按钮,添加两个按钮的响应函数:

  1. //向串口写入数据
  2. void CRS485CommDlg::OnSend()
  3. {
  4. // TODO: Add your control notification handler code here
  5. OVERLAPPED m_osWrite;
  6. memset(&m_osWrite,0,sizeof(OVERLAPPED));
  7. m_osWrite.hEvent=CreateEvent(NULL,TRUE,FALSE,NULL);
  8. char lpOutBuffer[7];
  9. memset(lpOutBuffer,''\0'',7);
  10. lpOutBuffer[0]=''\x11'';
  11. lpOutBuffer[1]=''0'';
  12. lpOutBuffer[2]=''0'';
  13. lpOutBuffer[3]=''1'';
  14. lpOutBuffer[4]=''0'';
  15. lpOutBuffer[5]=''1'';
  16. lpOutBuffer[6]=''\x03'';
  17. DWORD dwBytesWrite=7;
  18. COMSTAT ComStat;
  19. DWORD dwErrorFlags;
  20. BOOL bWriteStat;
  21. ClearCommError(hCom,&dwErrorFlags,&ComStat);
  22. bWriteStat=WriteFile(hCom,lpOutBuffer,dwBytesWrite,& dwBytesWrite,&m_osWrite);
  23. if(!bWriteStat)
  24. {
  25. if(GetLastError()==ERROR_IO_PENDING)
  26. {
  27. WaitForSingleObject(m_osWrite.hEvent,1000);
  28. }
  29. }
  30. }
  31. //接受串口信息
  32. void CRS485CommDlg::OnReceive()
  33. {
  34. // TODO: Add your control notification handler code here
  35. OVERLAPPED m_osRead;
  36. memset(&m_osRead,0,sizeof(OVERLAPPED));
  37. m_osRead.hEvent=CreateEvent(NULL,TRUE,FALSE,NULL);
  38. COMSTAT ComStat;
  39. DWORD dwErrorFlags;
  40. char str[100];
  41. memset(str,''\0'',100);
  42. DWORD dwBytesRead=100;//读取的字节数
  43. BOOL bReadStat;
  44. ClearCommError(hCom,&dwErrorFlags,&ComStat);
  45. dwBytesRead=min(dwBytesRead, (DWORD)ComStat.cbInQue);
  46. bReadStat=ReadFile(hCom,str,dwBytesRead,&dwBytesRead,&m_osRead);
  47. if(!bReadStat)
  48. {
  49. if(GetLastError()==ERROR_IO_PENDING)//GetLastError()函数返回ERROR_IO_PENDING,表明串口正在进行读操作
  50. {
  51. WaitForSingleObject(m_osRead.hEvent,2000);
  52. //使用WaitForSingleObject函数等待,直到读操作完成或延时已达到2秒钟
  53. //当串口读操作进行完毕后,m_osRead的hEvent事件会变为有信号
  54. }
  55. }
  56. PurgeComm(hCom,PURGE_TXABORT|PURGE_RXABORT|PURGE_TXCLEAR|PURGE_RXCLEAR);
  57. m_disp=str;
  58. UpdateData(FALSE);
  59. }

4. 关闭串口

  打开ClassWizard,为静态文本框IDC_DISP添加CString类型变量m_disp,同时添加WM_CLOSE的相应函数:

  1. void CRS485CommDlg::OnClose()
  2. { // TODO: Add your message handler code here and/or call default CloseHandle(hCom); //程序退出时关闭串口 CDialog::OnClose(); }

发表评论

表情:
评论列表 (有 0 条评论,279人围观)

还没有评论,来说两句吧...

相关阅读

    相关 VC串口编程实例2

      本例程采用异步串口操作,我们只介绍软件部分,RS485接口接线方法不作介绍,感兴趣的读者可以查阅相关资料。   打开VC++6.0,新建基于对话框的工程RS485Com

    相关 VC串口编程实例1

      本例程采用同步串口操作,我们只介绍软件部分,RS485接口接线方法不作介绍,感兴趣的读者可以查阅相关资料。   打开VC++6.0,新建基于对话框的工程RS485Comm

    相关 VC串口API通信详解

      在工业控制中,工控机(一般都基于Windows平台)经常需要与智能仪表通过串口进行通信。串口通信方便易行,应用广泛。   一般情况下,工控机和各智能仪表通过RS485总

    相关 VC多线程编程实例

    在本文中我们以实例的形式来介绍一下VC中的多线程编程,本文中没有涉及线程的相关同步问题我们更多的是讲述VC下的线程相关操作与编写方法等,至于线程的同步问题请参见《[WinCE线