一个串口通信的类

CSerial.h

// Serial.h
 
#define FC_DTRDSR       0x01
#define FC_RTSCTS       0x02
#define FC_XONXOFF      0x04
#define ASCII_BEL       0x07
#define ASCII_BS        0x08
#define ASCII_LF        0x0A
#define ASCII_CR        0x0D
#define ASCII_XON       0x11
#define ASCII_XOFF      0x13
// #define COMMTIMERUP     5000 
 
class CSerial  
{
public:
    CSerial();
    virtual ~CSerial();
 
    BOOL Open( int nPort, int nBaud);
    BOOL PortClose (void);
    BOOL Send(unsigned char *buff,int len);
    BOOL Revice(unsigned char *buff,int maxlen,int &len);
    void ClearBuff();
 
    BOOL m_bOpened;
    BOOL porterror;
    HANDLE m_hIDComDev;
    DWORD dwError;
    DWORD dwThreadID;
         
};
extern CSerial mySerial;



CSerial.cpp

// Serial.cpp: implementation of the CSerial class.
//
//
 
#include "stdafx.h"
#include "resource.h"
#include "CSerial.h"
 
 
//
// Construction/Destruction
//
 
 
    
CSerial::CSerial()
{
   m_hIDComDev = NULL;
   m_bOpened = FALSE;
   porterror =FALSE;
}
 
CSerial::~CSerial()
{
  PortClose ();
}
 
BOOL CSerial::Open( int nPort, int nBaud )
{
 
    if( m_bOpened ) return( TRUE );
 
    char szPort[15];
    DCB dcb;
 
    sprintf( szPort, _T("COM%d:"), nPort );
    m_hIDComDev = CreateFile( szPort,//指向端口名的Pointer 
        GENERIC_READ | GENERIC_WRITE,//允许读和写
        0,//独占方式(共享模式)
        NULL, //NULL为缺省安全属性
        OPEN_EXISTING,//打开而不是创建(创建方式)
        0,//FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED,//重叠方式(文件属性和标志)
        NULL 
        );
 
    if( m_hIDComDev == INVALID_HANDLE_VALUE)
    {
        DWORD dwError=GetLastError();
   ::MessageBox(0,_T("打开端口出错!请确定端口设备的使用情况后关闭重试."),_T("ERROR!"),MB_OK );
        porterror=TRUE;
        return( FALSE );
    }
    else
    {
//   ::MessageBox(0,_T("打开端口正确!"),_T("RIGHT!"),MB_OK );
     porterror=FALSE;
     m_bOpened = TRUE;
    }
/*
 
    memset( &m_OverlappedRead, 0, sizeof( OVERLAPPED ) );
     memset( &m_OverlappedWrite, 0, sizeof( OVERLAPPED ) );
*/
  //
  // Change the CommTimeOuts structure settings.
    COMMTIMEOUTS CommTimeOuts;
    GetCommTimeouts (szPort, &CommTimeOuts);
 
    CommTimeOuts.ReadIntervalTimeout =MAXWORD;          // not use
    CommTimeOuts.ReadTotalTimeoutMultiplier = 5;  // 5*byte
    CommTimeOuts.ReadTotalTimeoutConstant = 10000;   //5s 
    CommTimeOuts.WriteTotalTimeoutMultiplier = 1;  //5*byte
    CommTimeOuts.WriteTotalTimeoutConstant = 1;   //5s 
 
    if(!SetCommTimeouts( m_hIDComDev, &CommTimeOuts ))
    {
        //could not creatthe read thread;
        ::MessageBox (NULL,_T("不能设置延时参数"),_T("ERROR"),MB_OK);
         DWORD dwError=GetLastError();
         return FALSE;
    }
/*    else
    {
       ::MessageBox (NULL,_T("延时参数设置正确"),_T("RIGHT"),MB_OK);
    }
 
    wsprintf( szComParams, "COM%d:%d,n,8,1", nPort, nBaud );
 
    m_OverlappedRead.hEvent = CreateEvent( NULL, TRUE, FALSE, NULL );
    m_OverlappedWrite.hEvent = CreateEvent( NULL, TRUE, FALSE, NULL );
*/
/端口设置
    dcb.DCBlength = sizeof( DCB );
    //获得端口设置信息
    GetCommState( m_hIDComDev, &dcb );
    //端口波特率
    dcb.BaudRate = nBaud;
 
    dcb.fBinary = TRUE;               // Binary mode; no EOF check 
    dcb.fParity = FALSE;               // Enable parity checking 
    dcb.fOutxCtsFlow = FALSE;         // No CTS output flow control ***keiy**
    dcb.fOutxDsrFlow = FALSE;         // No DSR output flow control ****keiy*
    dcb.fDtrControl = DTR_CONTROL_ENABLE; 
                                      // DTR flow control type 
    dcb.fDsrSensitivity = FALSE;      // DSR sensitivity 
    dcb.fTXContinueOnXoff = FALSE;     // XOFF continues Tx 
    dcb.fOutX = FALSE;                // No XON/XOFF out flow control 
    dcb.fInX = FALSE;                 // No XON/XOFF in flow control 
    dcb.fErrorChar = FALSE;           // Disable error replacement 
    dcb.fNull = FALSE;                // Disable null stripping 
    dcb.fRtsControl = RTS_CONTROL_ENABLE; 
                                      // RTS flow control 
    dcb.fAbortOnError = FALSE;        // Do not abort reads/writes on 
                                      // error
    dcb.ByteSize = 8;                 // Number of bits/byte, 4-8 
    dcb.Parity = 0;            // 0-4=no,odd,even,mark,space 
    dcb.StopBits = ONESTOPBIT;        // 0,1,2 = 1, 1.5, 2 
 
/*    unsigned char ucSet;
    ucSet = (unsigned char) ( ( FC_RTSCTS & FC_DTRDSR ) != 0 );
    ucSet = (unsigned char) ( ( FC_RTSCTS & FC_RTSCTS ) != 0 );
    ucSet = (unsigned char) ( ( FC_RTSCTS & FC_XONXOFF ) != 0 );*/
 
 
    if( !SetCommState( m_hIDComDev, &dcb )||
        !SetupComm( m_hIDComDev, 1024, 1024 )/*||
        m_OverlappedRead.hEvent == NULL ||
        m_OverlappedWrite.hEvent == NULL*/ )
    {
        DWORD dwError = GetLastError();
        //if( m_OverlappedRead.hEvent != NULL ) CloseHandle( m_OverlappedRead.hEvent );
        //if( m_OverlappedWrite.hEvent != NULL ) CloseHandle( m_OverlappedWrite.hEvent );
        CloseHandle( m_hIDComDev );
        //::MessageBox(0,_T("初始化端口不正确!"),_T("ERROR!"),MB_OK );
        return( FALSE );
        }
 
else
{//::MessageBox (NULL,_T("初始化正确,欢迎使用终端测试仪!"),_T("RIGHT"),MB_OK);
 
}
 
    m_bOpened = TRUE;
 
    return( m_bOpened );
 
}
 
//======================
BOOL CSerial::PortClose (void)
{
    if (m_bOpened){
    // Close the communication port.
    if (!CloseHandle (m_hIDComDev))
    {
     DWORD dwError = GetLastError ();
     ::MessageBox (NULL,_T("关闭端口错误"),_T("ERROR"),MB_OK);
    }
     else  {
//        ::MessageBox (NULL,_T("正常退出,谢谢使用!"),_T("RIGHT"),MB_OK); 
        m_bOpened=FALSE;
     }
    }
    else{return(FALSE); }
return( TRUE );  
}
 
BOOL CSerial::Send(unsigned char *buff,int len)
{
  DWORD wd=0;
  if (m_bOpened)
       return WriteFile(m_hIDComDev,buff,len,&wd,NULL);
  else
      return false;
}
 
//BOOL   timeup=FALSE;
 
//void TimerUp()
//{
//   timeup=TRUE;
//}
BOOL CSerial::Revice(unsigned char *buff,int maxlen,int &len)
{
//  DWORD ev,rv;
//  int mytimerid;
 
    // 清空接收   
  buff[len=0]='\x0';
 
  if ((!m_bOpened) || maxlen==0)
      return false;
    //  设定超时
//  mytimerid=SetTimer(NULL,1,COMMTIMERUP,(TIMERPROC)TimerUp);
//  timeup=FALSE;
   
      // 指定监视的端口事件
//..//  SetCommMask (m_hIDComDev, EV_RXCHAR | EV_CTS | EV_DSR | EV_RLSD );
 // while (!timeup) // 总时间 5S
//  {
    // 等待端口事件
//..//    WaitCommEvent (m_hIDComDev, &ev, 0);
 
    // 重新指定监视的端口事件
//..//    SetCommMask (m_hIDComDev, EV_RXCHAR | EV_CTS | EV_DSR );
 
//..//    if (ev & EV_RXCHAR) 
//    {
        ReadFile (m_hIDComDev,buff, maxlen, (DWORD *)&len, 0);
 
      if (len!=0)
        {
//               KillTimer(NULL,mytimerid);
               //len=rv;
               buff[len]=0;
               return true;   
        }
//    }
    // 等。。。
 //  MSG msg;
//    if(PeekMessage((LPMSG)&msg,(HWND)NULL,(WORD)NULL,(WORD)NULL,TRUE))
//    {
//        TranslateMessage((LPMSG)&msg);
//        DispatchMessage((LPMSG)&msg);
//    }
     
//  }
//  buff[len]='\x0';
//  KillTimer(NULL,mytimerid);
  return false;
}
 
void CSerial::ClearBuff()
{
    PurgeComm(m_hIDComDev,PURGE_RXCLEAR);
}
 
CSerial mySerial;


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: MFC 串口通信的例子:1. 首先,在MFC应用程序中,添加一个串口,将要使用的串口设备映射到中。2. 然后,在MFC应用程序中调用CreateFile()函数来创建串口设备句柄,并指定COM端口号。3. 之后,调用SetupComm()函数来设置串口设备的缓冲区大小。4. 接下来,调用GetCommState()函数来获取串口设备的当前状态,在此基础上设置新的状态,并调用SetCommState()函数设置新的状态。5. 然后,调用SetCommMask()函数,设置串口设备支持的事件,并调用WaitCommEvent()函数等待事件发生。6. 最后,调用ReadFile()函数读取串口设备发送的数据,调用WriteFile()函数向串口设备发送数据,并调用CloseHandle()函数关闭串口设备句柄。 ### 回答2: MFC(Microsoft Foundation Classes)是一种用于Windows操作系统的C++应用程序框架,可用于编写Windows图形用户界面(GUI)应用程序。下面是一个基于MFC串口通信示例的简要描述: 1. 首先,在MFC应用程序中创建一个新的对话框或窗口,并添加一个串口控制按钮和一个用于接收数据的文本框。 2. 在对话框的头文件中,声明一个CSerialPort对象来执行串口通信操作,例如:CSerialPort m_serialPort; 3. 在对话框的OnInitDialog()函数中,初始化串口通信,设置串口参数和事件处理函数。例如: - 调用m_serialPort.InitPort()来初始化串口。 - 调用m_serialPort.SetPort()来设置串口号。 - 调用m_serialPort.SetBaudRate()来设置波特率。 - 调用m_serialPort.SetParity()和m_serialPort.SetStopBits()来设置奇偶校验位和停止位。 - 调用m_serialPort.SetEvent()来设置串口事件。 - 调用m_serialPort.OpenListenThread()打开串口线程。 4. 在按钮的点击事件处理函数中,调用CSerialPort的WriteToPort()函数将数据发送到串口。 5. 实现串口消息处理函数OnReceive(),可以在其中读取串口接收到的数据,并显示到文本框上。例如: - 调用m_serialPort.ReadData()来读取串口数据。 - 将读取到的数据展示在文本框上,可以使用UpdateData()函数。 6. 在程序结束时,关闭串口连接。例如,在OnDestroy()函数中调用m_serialPort.ClosePort()关闭串口。 通过上述步骤,你可以使用MFC编写一个简单的串口通信示例。当你点击发送按钮时,数据将被发送到串口,然后通过串口接收线程读取数据,最后将数据显示在文本框上。这个例子可以帮助你了解如何使用MFC进行串口通信。 ### 回答3: MFC(Microsoft Foundation Class)是一个用于开发Windows应用程序的框架,它提供了一系列的和函数来简化应用程序的开发。在MFC中,我们可以使用串口通信来实现两个设备之间的数据传输。 要在MFC中实现串口通信,我们可以按照以下步骤进行: 1. 首先,在MFC应用程序中创建一个新的对话框或文档/视图项目。 2. 在对话框或文档/视图中,添加一个按钮用来打开串口。在按钮的点击事件处理函数中调用CSerialPort的OpenPort()函数,该函数即用来打开串口连接。 3. 在对话框或文档/视图中添加一个文本框或编辑框,用来显示接收到的串口数据。 4. 在对话框或文档/视图中添加一个按钮用来关闭串口。在按钮的点击事件处理函数中调用CSerialPort的ClosePort()函数,该函数即用来关闭串口连接。 5. 在对话框或文档/视图中添加一个编辑框,用来输入需要发送的数据。 6. 在对话框或文档/视图中添加一个按钮用来发送数据。在按钮的点击事件处理函数中调用CSerialPort的SendData()函数,该函数用来发送数据到串口。 7. 在对话框或文档/视图中重写WM_COMMNOTIFY消息处理函数,该函数用来处理从串口接收到的数据。在该消息处理函数中,首先调用CSerialPort的ReadData()函数读取串口接收到的数据,然后将读取到的数据显示在文本框或编辑框中。 上述步骤简单介绍了在MFC中使用串口进行通信的基本操作流程。具体的实现过程可以参考MFC提供的文档或者在线教程。同时,在实际开发过程中,还需要注意串口参数的设置、错误处理、数据转换等细节问题,以保证串口通信的正常运行。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值