VS2015串口通信编程(MFC)笔记

前言:

一直很喜欢CSDN这个平台,这是第一次在这里写博客,内容若有不当之处还请多多指教,谢谢!

本笔记主要是记录一些开发过程中可能会遇到的一些注意事项,以加强记忆和提醒。

一、 开发目的:

  开发一个简单的上位机界面程序,实现上位机与下位机串口通信,从而对下位机进行程序升级。

二、 开发环境:

  win7系统; VS2015社区版(免费);USB转TTL模块*2(用于调试);

三、内容:

1.创建MFC项目工程,界面设计直接使用VS2015提供的控件,无非也就是一些按钮、编辑框、下拉框、文本、进度条。

这写可视化操作还是比较简单的。

2.语言细节:

(1). 移位运算: 使用 >> 时高位补1, 使用 << 时低位补0;

(2). int转CString: str.Format(_T("%d"), k); // int k; CString str;

(3) CString转int:k = _ttoi(str); //int k; CString str;

(4).取字符串中的某一部分:str.Mid(s, n); //str的第s个字符开始的n个字符(包括s在内)

(5).将unicode 转换为 char:

   iLength = WideCharToMultiByte(CP_ACP,0, FilePath, -1,NULL,0,NULL,NULL);//先取得长度

   WideCharToMultiByte(CP_ACP,0, FilePath, -1, FPchar, iLength,NULL, NULL);//转换后存入FPchar数组中


3. 选用MSCOMM控件方式进行串口通信

(1). 只能打开不大于COM16的串口;该控件不能在其他线程中直接被调用,但可以将其操作封装到函数中被间接使用。

(2). 接收:MSCOMM控件中断事件,用于接收下位机消息。

(3). 发送:另外创建一个子线程用于控制发送流程。

m_pThread = AfxBeginThread(UpgradeThread, (LPVOID)(this), THREAD_PRIORITY_BELOW_NORMAL, 256 * 1024);//stack size = 256*1024


4. 选用API函数方式进行串口通信

(1). 使用API函数CreateFile打开串口时传入的串口号参数前面需加“\\\\.\\”,否则无法打开大于COM9的串口。(例如打开COM10: "\\\\.\\COM10");

主要有四个函数:

CreateFile //打开

CloseHandle //关闭

ReadFile  //读取

WriteFile  //写入

主要三个变量: 

HANDLE m_HdlSerial = INVALID_HANDLE_VALUE;  //串口句柄

OVERLAPPED m_osRead; //用于读取控制

OVERLAPPED m_osWrite; //用于发送控制

(2). 打开串口(异步重叠方式)

//异步重叠,8个数据位,无校验,1个停止位

HANDLE KSerialOpen(LPCTSTR COMx, DWORD baudrate)

{

HANDLE hCom;

BOOL state;

DCB SerialDCB;


//打开

hCom = CreateFile(

COMx,

GENERIC_READ | GENERIC_WRITE,

0,

NULL,

OPEN_EXISTING,

FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED,

NULL

);

if (hCom == INVALID_HANDLE_VALUE) {//打开失败

return INVALID_HANDLE_VALUE;

}

state = GetCommState(hCom, &SerialDCB);

SerialDCB.BaudRate = baudrate; /* Baudrate at which running, CBR_115200 */

SerialDCB.ByteSize = 8;        /* Number of bits/byte, 4-8 */

SerialDCB.Parity = NOPARITY;    /* 0-4=None,Odd,Even,Mark,Space */

SerialDCB.StopBits = ONESTOPBIT; /* 0,1,2 = 1, 1.5, 2 */

state = SetCommState(hCom, &SerialDCB);

if (!state) { 

CloseHandle(hCom);

return INVALID_HANDLE_VALUE; 

}

//设置缓冲区大小

state = SetupComm(hCom, SERIAL_RECV_BUFFER, SERIAL_SEND_BUFFER);//扱˙, ‰»Îª∫≥Â¥Û–°£¨ ‰≥ˆª∫≥Â¥Û–°

if (!state) {

CloseHandle(hCom);

return INVALID_HANDLE_VALUE;

}

PurgeComm(hCom,  PURGE_TXCLEAR | PURGE_RXCLEAR | PURGE_TXABORT | PURGE_RXABORT);

return hCom;

}


(3). 接收:创建一个线程用于接收下位机消息。

//接收线程

UINT ComRxThread(LPVOID pParam)

{

CmyFileMissionDlg *pMainDlg = (CmyFileMissionDlg *)pParam;


DWORD dwBytes;

int iReadBytes, res;

DWORD dwError = 0;

DWORD dwEvtMask = 0;

unsigned char RxBuffer[1024];


//设置接收时间掩码

if (SetCommMask(m_HdlSerial, EV_RXCHAR) == 0) {

AfxMessageBox(_T("…Ë÷√ ¬º˛—¬Î ß∞‹!"));

return 1;

}

while (TRUE

        {

    if (m_HdlSerial == INVALID_HANDLE_VALUE) {

        return 0;

    }

    if (WaitCommEvent(m_HdlSerial, &dwEvtMask, &m_osRead)) {

        if (dwEvtMask & EV_RXCHAR) { //接收事件

    //TODO

    res = WaitForSingleObject(m_osRead.hEvent, INFINITE);//一直等到异步处理完成

    if (WAIT_OBJECT_0 == res) {

//TODO:进行读取及其他应用操作 (ReadFile)

    }

    PurgeComm(m_HdlSerial, PURGE_RXABORT | PURGE_RXCLEAR);

        }

    }

        }

return 0;

}


(4). 发送:创建一个线程用于控制发送流程。

//发送函数

int KSerialWrite(HANDLE hCom, LPOVERLAPPED lpOverlTx, unsigned char *Buffer, int cnt)

{

BOOL bStatus;

DWORD dwErrorFlags;

DWORD dwBytes = 0;

COMSTAT ComStat;

ClearCommError(hCom, &dwErrorFlags, &ComStat);

bStatus = WriteFile(hCom, Buffer, cnt, &dwBytes, lpOverlTx);

if (!bStatus) {

if (GetLastError() == ERROR_IO_PENDING) {//

GetOverlappedResult(hCom, lpOverlTx, &dwBytes, TRUE);

} else {

dwBytes = 0;

}

}

return dwBytes;

}


5. 关于实现文件选择  

//选择文件,并将文件名显示在编辑框中

void CmyFileMissionDlg::OnBnClickedButton2()

{

    CString fileName = _T("");//

    CString filter = _T("Œƒº˛ (*.bin; *.hex; *.txt)|*.bin;*.hex;*.txt||");//默认文件类型

    CString defaultDir;

    TCHAR exeDirectory[MAX_PATH];

    GetCurrentDirectory(MAX_PATH, exeDirectory);//获取当前路径作为默认路径

    defaultDir.Format(_T("%s"),exeDirectory);

    /* 打开方式打开 */

    CFileDialog openFileDlg(TRUE, defaultDir, fileName, OFN_HIDEREADONLY|OFN_READONLY, filter,NULL);

    openFileDlg.GetOFN().lpstrInitialDir = defaultDir;

    INT_PTR result = openFileDlg.DoModal();//选择文件

    if(result == IDOK) {//选择确定后,将文件名显示到 m_edit1 编辑框中

       m_edit1.SetWindowTextW(openFileDlg.GetPathName());

    }

}


6.后话:

由于本人主要是作嵌入式程序开发,对于上位机程序一般是作为辅助工具,主要用于开发一些简单调试工具。

对于VC++编程还只是接触皮毛,所以本文难免会有见解有误的地方,但是本着相互交流学习的态度还是毅然拙笔狂飞。

结尾之处,应该有:谢谢!






阅读更多
想对作者说点什么? 我来说一句

串口通信编程详解和实例

2008年08月08日 42KB 下载

VS2017-MFC串口通信(基于CSerialPort类)

2018年04月24日 134.75MB 下载

VS串口通信程序

2017年12月28日 67.11MB 下载

没有更多推荐了,返回首页

关闭
关闭