我用CreateFile函数异步打开串口,然后设置了他的波特率等参数,用ReadFileEx(handle,pBuffer,sizeof(pBuffer),&ReadNum,&rOverLaped,(LPOVERLAPPED_COMPLETION_ROUTINE)lpCompletionRoutine)函数来读串口,
其中初始化定义 OVERLAPPED rOverLaped = {0},定义void lpCompletionRoutine(){};
函数返回的是true,但是ReadNum为0,也就是没有读到数据,GetLastError()也返回0,但没有进入lpCompletionRoutine函数内部执行,
另外,我是用线程里来读串口的,但是这个线程调用读函数一遍后,就不再进入读函数了,是不是线程阻塞之类的,还需要设置什么等待信号之类的吗?

解决方案 »

  1.   

    参考一下,看看那里设错了 .h
    public:
    HANDLE hCom; //串口句柄
    HANDLE hThread; //线程句柄
    DWORD dwThreadID; //线程ID
    char DataBuffer[1024]; //接收数据的缓冲区
    CString strPath;
    CString strText;  .cpp
    //自定义一个消息
    const CM_RECEIVE = WM_USER+100;
    OVERLAPPED tOverLaped= {0}; //线程函数使用的OVERLAPPED结构
    OVERLAPPED wOverLaped = {0}; //写操作使用的OVERLAPPED结构
    OVERLAPPED rOverLaped = {0}; //读操作使用的OVERLAPPED结构
    BOOL IsFun = TRUE; //线程是否运行
    BOOL IsStop = FALSE; //数据是否发送完毕
    ---------------------------------------------------------------------------------------------------------OpenPort()//打开串口
    {
    hCom = CreateFile("COM1", //打开串口1
    GENERIC_READ | GENERIC_WRITE, //允许读和写操作
    0, //独占方式
    NULL,
    OPEN_EXISTING, //打开一个存在的串口
    FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, //异步方式打开
    NULL
    );if(hCom == INVALID_HANDLE_VALUE)
    {
    MessageBox("端口打开失败");
    }
    }
    ---------------------------------------------------------------------------------------------------------
    OnSetport()//设置串口  
    {
      SetupComm(hCom, 1024, 1024); //设置发送和接收缓冲区大小COMMTIMEOUTS TimeOuts;  
    // 设定读超时   
    TimeOuts.ReadIntervalTimeout = MAXDWORD;   
    TimeOuts.ReadTotalTimeoutMultiplier = 0;   
    TimeOuts.ReadTotalTimeoutConstant = 0;// 设置写超时   
    TimeOuts.WriteTotalTimeoutMultiplier = 100;   
    TimeOuts.WriteTotalTimeoutConstant = 500;   
    SetCommTimeouts(hCom, &TimeOuts);  //设置串口信息
    DCB dcb;
    GetCommState(hCom, &dcb);//获取串口当前配置参数
    dcb.BaudRate = 9600;//波特率
    dcb.fBinary = TRUE; //允许二进制传输
    dcb.fParity = TRUE;//允许奇偶校验
    dcb.ByteSize = 8;//数据位
    dcb.Parity = NOPARITY;//奇校验
    dcb.StopBits = ONESTOPBIT;//1个停止位
    if(!SetCommState(hCom,&dcb))
    {
    MessageBox("设置失败", "提示");
    return; 
    }
    if(!SetCommMask(hCom, EV_RXCHAR | EV_TXEMPTY))
    {
    MessageBox("掩码设置失败", "提示");
    return;
    }
    PurgeComm(hCom, PURGE_TXCLEAR|PURGE_RXCLEAR);DWORD param;
    hThread = CreateThread((LPSECURITY_ATTRIBUTES)NULL, //安全属性
    0, //初始化线程栈的大小,缺省为与主线程大小相同
    (LPTHREAD_START_ROUTINE)ThreadFunction, //线程的全局函数
    &param, //此处传入了主框架的句柄  
    0, //创建后立即激活
    &dwThreadID); //保存新线程的id
    if (hThread == INVALID_HANDLE_VALUE)
    {
    MessageBox("线程创建失败", "提示");
    return;
    }
    }
    -----------------------------------------------------------------------------------------------------
    DWORD ThreadFunction(LPVOID pParam)//线程函数
    {
    DWORD dwEvtMask ,dwResult;
    // tOverLaped.Internal=0;   
    // tOverLaped.InternalHigh=0;   
      tOverLaped.Offset=0;   
      tOverLaped.OffsetHigh=0;  
    tOverLaped.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);//创建一个事件
    while(IsFun)
    {
    //等待窗口事件
    WaitCommEvent(((CMSGCOMDlg*)AfxGetMainWnd())->hCom,
    &dwEvtMask, &tOverLaped);
    //如果事件没有信号,延时0.1秒
    dwResult = WaitForSingleObject(tOverLaped.hEvent, 100);if(dwResult == WAIT_OBJECT_0) //事件对象有信号
    {
    if(dwEvtMask == EV_RXCHAR) //接收到数据
    {
    if(IsStop)//发送停止
    {
    IsStop = FALSE;
    //发送消息,由消息处理函数接收数据
    ::PostMessage(AfxGetMainWnd()->m_hWnd, CM_RECEIVE,
    0,(LPARAM)EV_RXCHAR);


    }
    }
    return 0;
    }
    -----------------------------------------------------------------------------------------------------OnSend() //发送指令
    {
      DWORD res;
    DWORD factdata = 0;
    CString str;
    BOOL bwrite;
    m_data.GetWindowText(str);
    str += "\r";wOverLaped.Offset=0;   
    wOverLaped.OffsetHigh=0;
    wOverLaped.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); //创建一个事件对象
    //IsStop = FALSE;
    bwrite = WriteFile(hCom, str, str.GetLength(), &factdata, &wOverLaped);
    if (bwrite)
    {
    IsStop = TRUE;
    MessageBox("发送成功");
    }
    else if(GetLastError()==ERROR_IO_PENDING)
    {
    res = WaitForSingleObject(wOverLaped.hEvent, 2000); //延时2秒,等待数据发送
    //如果事件处于有信号状态,表示发送完成
    if (WAIT_OBJECT_0 == res) IsStop = TRUE; 
    else IsStop = FALSE;Sleep(1000); //延时1秒钟}
    }
    -------------------------------------------------------------------------------------------------------
    OnRecieveData(WPARAM wParam, LPARAM lParam)//接收数据
    {
    DWORD res, factbyte;
    DWORD dwBytesRead = 1024;
    char DataBuffer[1024]; //初始化数据缓冲区
    COMSTAT rst;
    ClearCommError(hCom, &res, &rst); //清空串口错误标志,记录当前通信状态
    DWORD S=rst.cbInQue;
    dwBytesRead = min(dwBytesRead, (DWORD)rst.cbInQue);
    // rOverLaped.Internal=0;   
    // rOverLaped.InternalHigh=0;   
    rOverLaped.Offset=0;   
    rOverLaped.OffsetHigh=0;
    rOverLaped.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL); //创建一个事件对象if (ReadFile(hCom, DataBuffer, dwBytesRead, &factbyte, &rOverLaped)) //读取数据到缓冲区中
    {
    DataBuffer[rst.cbInQue] = 0;
    IsStop = FALSE;
    }
    else

    //GetLastError()函数返回ERROR_IO_PENDING,表明串口正在进行读操作
    if(GetLastError()==ERROR_IO_PENDING)
    {
    res = WaitForSingleObject(rOverLaped.hEvent, 2000); 
    if (WAIT_OBJECT_0 == res) IsStop = FALSE; 

    }
    PurgeComm(hCom, PURGE_TXABORT | PURGE_RXABORT | PURGE_TXCLEAR | PURGE_RXCLEAR ); //清空输入、输出缓冲区  
    //IsStop = FALSE;
    CString str = DataBuffer;m_outwnd += str + "\r\n"; //输出到编辑框
    UpdateData(FALSE);
    }
      

  2.   

    执行ThreadFunction()函数时,进不到if(dwResult == WAIT_OBJECT_0)里面,是什么原因?直接出来主界面了
      

  3.   

    这时ReadFile()函数还没调用,没有读完是指没有读完连接串口的机器发送的所有数据还是没有读完填满缓冲区的数据,连接串口的机器是不停的发送数据,那我怎样才能读到数据?