1从零开始学VC系列教程之三.串口通信与自定义消息课程之前:首先请大家确认一下前面两章都已经熟悉,因为一些前面已经介绍过的基础操作在这里将不再详细说明,如果有什么问题,可以翻看一下前面的两章或者留言提问.本章是基于PC机与单片机的串口通信,用到了一个动态链接库和一个自定义消息.学习目标:掌握VC下串口编程式的方法,掌握动态库的静态调用及自定义消息.课程详解:1.参照第一章新建一个基于对话框的Vc工程,名称定义为Eg03.2.工程建立后,在对话框上加入一个组合框(ComboBox),ID号改为IDC_COMPORT用于选择使用PC机上的哪一个串口.在组合框后加入一个按钮,标题(Caption)改为”打开”,ID号改为IDC_BTN_PORTOPEN用于打开串口,开始通信.下面加入一个编程框(EDIT),ID号改为IDC_EDIT_RECMSG用于显示接收到的数据.在编程框下面再添加一个编程框(EDIT),.ID号改为IDC_EDIT_SEDMSG用于添加要发送的数据.然后在这个编程框后加入一个按钮.标题(Caption)为”发送”,ID号为IDC_BTN_SEND最后调整位置及大小如下23.添加Lib文件.这里介绍的串口通信用的不是VC自带的MSCOMM控件.原因有两个,一是顺便介绍一下动态库和自定义消息的用法.二是MSCOMM控件使用时数据类型转换比较复杂,并且使用也不是很方便.当然,以后也会介绍多线程串口通信给大家,我们会在后面开设一章多线程编程方法,并在那里详细介绍基于多线程的串口通信.这里使用一个动态库,其实也是别人封装好了的多线程通信,名字是Pcomm.在工程下载中,给出了三个文件,分别是Pcomm.h,Pcomm.lib,Pcomm.dll,现在请大家把这三个文件拷到工程目录,也就是Eg03这个文件夹中.至于什么是动态库,这三个文件倒底是什么作用,我们做完这个例程后再解释,现在还是先按步就搬,营造一个感性认识.下面添加Lib文件到工程.首先点击[工程](Project),选择下拉式菜单中的[设置](ProjectSettings)3然后会弹出一个对话框,在标签卡中选择[连接]然后在[对象/模块]中添加Pcomm.lib,完成后如上面所示,单击[确定]退出.这样,我们就为Pcomm.dll这个动态库添加了静态链接,同时,这也就是动态库的静态链接方法,当然,还有一步就是包含Pcomm.h这个头文件.在刚才的步骤中,我们将4Pcomm.Lib添加到工程,这个文件主要用于指定Pcomm.dll中各个功能函数的入口及地址,Pcomm.Lib就像一个地图指出目的地的路标,而真正的函数是在Pcomm.Dll中的.当然,为了方便调用,我们还要得到Pcomm.Dll中的函数声明,这些函数声明就在Pcomm.h这个头文件中,所以,大家打开Pcomm.h这个文件,只有函数及变量定义,并没有函数过程.下面我们来添加这个文件.4.打开左边的[工作空间](WorkSpace)中选择标签[ClassView](这里大家只能看到[Class…]这一步前两章已经详细介绍过了,大家可以参考.),然后双击[OnInitDialog]就可以打开代码窗口了,在原有头文件包含后面加入串口头文件引用.输入#include”Pcomm.h”就可以了,完成后如下图5这一步我们加入了动态库的函数声明,后面就可以直接使用Pcomm.Dll中的函数了.下面我们来添加事件响应.单击工作空间中间的标签[ResourceView](大家看到的是[Reso…]),再双击[IDD_EG03_DIALOG]就可以回到控件编辑状态.6首先为[打开]按钮添加代码.双击[打开]按钮,然后在按钮事件中添加.完成后如下voidCEg03Dlg::OnBtnPortopen(){//TODO:AddyourcontrolnotificationhandlercodeherePort=GetDlgItemInt(IDC_COMPORT);7if(SIO_OK!=sio_open(Port)){MessageBox(串口打开错误);}else{sio_ioctl(Port,BaudRate,DataBits|StopBits|Parity);sio_cnt_irq(Port,CntIrq,1);}}其中,sio开头的变量及函数都是Pcomm中的,我们来解释一下.sio_open是打开某个串口,传入的参数是串口号,如果我们要打开COM1,可以用sio_open(1),返回的参数在Pcomm里面定义了,如果返回SIO_OK就表示串口打开没有问题,否则,就是打开串口失败.sio_ioctl用于设置通信的相关信息,Port中串口号,BaudRate是波特率,DataBits是数据位数,StopBits是停止位数,Parity是校验.sio_cnt_irq用于设定中断回调函数.中断回调函数其实前面的Timer定时器里也提到过,在这里,我们设定一个中断回调函数,每当串口接收到指定字节数据时,系统就会自动调用这个中断回调函数,就像单片机中的串口中断函数一样.这里的回调函数名是CntIrq,我们将在后面定义.细心的朋友一定会发现,BaudRate,DataBits|StopBits|Parity这些都没定义过啊?所以要定义一下这些变量.参照前面的加入文件的方法,在头文件引用下一行加入以下宏定义#defineBaudRateB57600//波特率#defineDataBitsBIT_8//数据位#defineParityP_NONE//效验位#defineStopBitsSTOP_1//停止位完成后如图8下面我们要定义sio_cnt_irq一般来说,中断回调函数并不写在类里面,我们添加后如下9///////////////////////////串口中断回调函数//////////////////////////////////VOIDCALLBACKCntIrq(intport){if(::AfxGetMainWnd()){if(::AfxGetMainWnd()-m_hWnd){::PostMessage(::AfxGetMainWnd()-m_hWnd,WM_PCOMM,0,0);}}}学习过前面两章我们知道,这个中断回调函数只做了一件事情,就是发送一个WM_PCOMM消息到窗口.AfxGetMainWnd()这个函数用于获得主窗口,返回类型是CWnd的指针,主窗体句柄我们是不知道的,用AfxGetMainWnd()-m_hWnd来10获得.这样,消息就可到发到主窗体了.有了这个函数,每当串口接收了数据,就会发一个消息到窗体.WM_PCOMM这个消息不是系统的,也不是Pcomm本身的,它是我们自定义的一个消息,怎么定义呢?我们在前面说的宏定义后面再加入一个定义#defineWM_PCOMMWM_USER+500//自定义消息WM_USER是一个消息地址,这个是系统定义好的,从这个地址开始可以自定义消息,我们把WM_PCOMM定义为WM_USER+500也就是说,我们定义的这个消息位于WM_USER后面的偏移500,当然,这只是个地址,与执行先后无关.这个偏移大家可以自己随便设,不与别的自定义消息冲突就行了.消息定义好了还要为消息添加关联.首先要定义一个消息响应函数,名字随便,我们这里取名为OnPcomm(),双击[工作空间]中的Ceg03Dlg就可以打开窗体的文头件,这里定义了Ceg03Dlg这个类,我们在类定义里面添加一个成员函数.afx_msgvoidOnPcomm();//这里是我们自定义的消息响应函数完成后如图此外,这里还顺便定义了一个变量,就是前面我们用到的Port用于记录打开的串口号.public:intPort;11位置就放在DECLARE_MESSAGE_MAP()的前面.函数声明就可以了.现在来添加函数体.双击[OnInitDialog()],然后在该文件的最后添加一个函数.写成如下形式.voidCEg03Dlg::OnPcomm(){charbuf[200];intend=sio_read(Port,buf,100);if(end){CStringa,b=;GetDlgItemText(IDC_EDIT_RECMSG,b);buf[end]=0;for(inti=0;iend;i++){a.Format(%X,(unsignedchar)buf[i]);b+=a;}SetDlgItemText(IDC_EDIT_RECMSG,b);}}12这一段其实不难理解,因为前面两章已经介绍过多次了.sio_read是Pcomm的函数,从串口读取数据用.GetDlgItemText(IDC_EDIT_RECMSG,b);用于读出以前的历史记录,这样每次发上来的数据都放在后面连接起来.end是返回的收到的数据个数.用十六进制形式显示出来.做完了上面一些,我们差一步就可以收到数据了.因为数据发上来后,底层响应,并调用了回调函数,在回调函数里面,发出一个消息WM_PCOMM虽然我们在后面定义了一个Pcomm()函数专门用于响应这个消息,但这个自定义消息并不是自动连接到Pcomm()的,需要添加一个消息影射才能使WM_PCOMM消息影射到Pcomm()函数.双击左边[工作空间](WorkSpace)中的DoDataExchange(CDataExchange*pDX)DoDataExchange这个函数的下面一般都这是用于定义消息影射的,将下面一段程序增加一行,完成后如下.13BEGIN_MESSAGE_MAP(CEg03Dlg,CDialog)//{{AFX_MSG_MAP(CEg03Dlg)ON_WM_SYSCOMMAND()ON_WM_PAINT()ON_WM_QUERYDRAGICON()ON_BN_CLICKED(IDC_BTN_PORTOPEN,OnBtnPortopen)//}}AFX_MSG_MAPON_MESSAGE(WM_PCOMM,OnPcomm)//这里是消息影射END_MESSAGE_MAP()完成以后就可以按F7编译一下,如果无误就可以接收到数据了.运行后,选择正确的串口号,按一下[打开]按钮就可以了.现在我们再来看看怎么发送数据回到控件编辑状态,双击[发送]按钮,为该按钮添加代码.voidCEg03Dlg::OnBtnSend(){//TODO:AddyourcontrolnotificationhandlercodehereCStringa;unsignedcharb=0;GetDlgItemText(IDC_EDIT_SEDMSG,a);//取得编辑框内所有文本a.MakeUpper();//全部转换为大写for(unsignedchari=0;ia.GetLength();i+=3){14if(a.GetAt(i)='A'&&a.GetAt(i)='Z')b=(a.GetAt(i)-55)*16;//判断填入的是字母还是数字,并把字符转换成十六进制数elseif(a.GetAt(i)='0'&&a.GetAt(i)='9')b=(a.GetAt(i)-0x30)*16;if(a.GetAt(i+1)='A'&&a.GetAt(i+1)='Z')b+=(a.GetAt(i+1)-55);elseif(a.GetAt(i+1)='0'&&a.GetAt(i+1)='9')b+=(a.GetAt(i+1)-0x30);sio_putch(Port,b);//发送}}这一段主要是把获得的编辑框内的字串转换成十六进制的数字,转换一个发送一个.Cstring类型以前已经提起来,应该际上是一个类,MakeUpper是一个成员函数,用于将字符串全部转成大写.GetAt也是一个成员函数,可以取出字符串中任意下标的字符.sio_putch用于发送一个字符.填写待发送数据的时候要注意,每两位中间空格一下.填入的是十六进制数据.下面再来总结一下静态方式调用