原创 【转】从零开始学VC系列教程 三. 串口通信及自定义消息

2008-9-27 23:29 4396 10 12 分类: 软件与OS

首先感谢飞雪浪子 QQ:570733482 的无私精神,谢谢。


帮他打个广告哈,呵呵  百强电子http://www.bqmcu.com.cn/


这个是原帖  http://www.ouravr.com/bbs/bbs_content.jsp?bbs_sn=1551606&bbs_id=9999


从零开始学VC系列教程之 三.串口通信与自定义消息
课程之前:首先请大家确认一下前面两章都已经熟悉,因为一些前面已经介绍过的基础操作在这里将不再详细说明,如果有什么问题,可以翻看一下前面的两章或者留言提问.本章是基于PC机与单片机的串口通信,用到了一个动态链接库和一个自定义消息.
学习目标:掌握VC下串口编程式的方法,掌握动态库的静态调用及自定义消息.
课程详解:
1.        参照第一章新建一个基于对话框的Vc工程,名称定义为Eg03.


点击看大图
图片01 (原文件名:001.jpg) 

 
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 最后调整位置及大小如下


ourdev_433330.jpg
图片02 (原文件名:002.jpg) 

 
3.        添加Lib文件.这里介绍的串口通信用的不是VC自带的MSCOMM控件.原因有两个,一是顺便介绍一下动态库和自定义消息的用法.二是MSCOMM控件使用时数据类型转换比较复杂,并且使用也不是很方便.当然,以后也会介绍多线程串口通信给大家,我们会在后面开设一章多线程编程方法,并在那里详细介绍基于多线程的串口通信.这里使用一个动态库,其实也是别人封装好了的多线程通信,名字是Pcomm.在工程下载中,给出了三个文件,分别是Pcomm.h, Pcomm.lib, Pcomm.dll,现在请大家把这三个文件拷到工程目录,也就是Eg03这个文件夹中.至于什么是动态库,这三个文件倒底是什么作用,我们做完这个例程后再解释,现在还是先按步就搬,营造一个感性认识.下面添加Lib文件到工程.首先点击[工程](Project),选择下拉式菜单中的[设置](ProjectSettings)


点击看大图
图片03 (原文件名:003.jpg) 

 
然后会弹出一个对话框,在标签卡中选择[连接]


点击看大图
图片04 (原文件名:004.jpg) 

 
然后在[对象/模块]中添加Pcomm.lib,完成后如上面所示,单击[确定]退出.这样,我们就为Pcomm.dll这个动态库添加了静态链接,同时,这也就是动态库的静态链接方法,当然,还有一步就是包含Pcomm.h这个头文件.在刚才的步骤中,我们将Pcomm.Lib添加到工程,这个文件主要用于指定Pcomm.dll中各个功能函数的入口及地址,Pcomm.Lib就像一个地图指出目的地的路标,而真正的函数是在Pcomm.Dll中的.当然,为了方便调用,我们还要得到Pcomm.Dll中的函数声明,这些函数声明就在Pcomm.h这个头文件中,所以,大家打开Pcomm.h这个文件,只有函数及变量定义,并没有函数过程.下面我们来添加这个文件.
4.        打开左边的[工作空间](WorkSpace)中选择标签[ClassView](这里大家只能看到[Class…] 这一步前两章已经详细介绍过了,大家可以参考.),然后双击[OnInitDialog]就可以打开代码窗口了,在原有头文件包含后面加入串口头文件引用.输入#include ”Pcomm.h”就可以了,完成后如下图


点击看大图
图片05 (原文件名:005.jpg) 

 
这一步我们加入了动态库的函数声明,后面就可以直接使用Pcomm.Dll中的函数了.下面我们来添加事件响应.
单击工作空间中间的标签[ResourceView](大家看到的是[Reso…]),再双击[IDD_EG03_DIALOG]就可以回到控件编辑状态.


点击看大图
图片06 (原文件名:006.jpg) 

 
首先为[打开]按钮添加代码.双击[打开]按钮,然后在按钮事件中添加.完成后如下
void CEg03Dlg::OnBtnPortopen() 
{
        // TODO: Add your control notification handler code here
        Port=GetDlgItemInt(IDC_COMPORT);
        if(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这些都没定义过啊?所以要定义一下这些变量.参照前面的加入文件的方法,在头文件引用下一行加入以下宏定义
#define BaudRate B57600 //波特率
#define DataBits BIT_8 //数据位
#define Parity P_NONE //效验位
#define StopBits STOP_1 //停止位
完成后如图


点击看大图
图片07 (原文件名:007.jpg) 

 
下面我们要定义sio_cnt_irq 一般来说,中断回调函数并不写在类里面,我们添加后如下


点击看大图
图片08 (原文件名:008.jpg) 

 
///////////////////////////串口中断回调函数//////////////////////////////////
VOID CALLBACK CntIrq(int port) 
{
        if(::AfxGetMainWnd()) 
{
if(::AfxGetMainWnd()->m_hWnd)
{
::PostMessage(::AfxGetMainWnd()->m_hWnd,WM_PCOMM,0,0); 
}
}
}
学习过前面两章我们知道,这个中断回调函数只做了一件事情,就是发送一个WM_PCOMM消息到窗口. AfxGetMainWnd()这个函数用于获得主窗口,返回类型是CWnd的指针,主窗体句柄我们是不知道的,用AfxGetMainWnd()->m_hWnd来获得.这样,消息就可到发到主窗体了.有了这个函数,每当串口接收了数据,就会发一个消息到窗体.WM_PCOMM这个消息不是系统的,也不是Pcomm本身的,它是我们自定义的一个消息,怎么定义呢?我们在前面说的宏定义后面再加入一个定义
#define WM_PCOMM WM_USER+500 //自定义消息


点击看大图
图片09 (原文件名:009.jpg) 

 
WM_USER是一个消息地址,这个是系统定义好的,从这个地址开始可以自定义消息, 我们把WM_PCOMM定义为WM_USER+500也就是说,我们定义的这个消息位于WM_USER后面的偏移500,当然,这只是个地址,与执行先后无关.这个偏移大家可以自己随便设,不与别的自定义消息冲突就行了.消息定义好了还要为消息添加关联.首先要定义一个消息响应函数,名字随便,我们这里取名为OnPcomm(),双击[工作空间]中的Ceg03Dlg就可以打开窗体的文头件,这里定义了Ceg03Dlg 这个类,我们在类定义里面添加一个成员函数.
afx_msg void OnPcomm(); //这里是我们自定义的消息响应函数
完成后如图
此外,这里还顺便定义了一个变量,就是前面我们用到的Port 用于记录打开的串口号.
public:
        int Port;


点击看大图
图片10 (原文件名:010.jpg) 

 
位置就放在DECLARE_MESSAGE_MAP() 的前面.函数声明就可以了.现在来添加函数体.双击[OnInitDialog( )],然后在该文件的最后添加一个函数.写成如下形式.
void CEg03Dlg::OnPcomm()
{
        char buf[200];
        int end=sio_read(Port,buf,100);
        if(end)
        {
                CString a,b="";
                GetDlgItemText(IDC_EDIT_RECMSG,b);
                buf[end]=0;
                for(int i=0;i<end;i++)
                {
                        a.Format("%X ",(unsigned char)buf);
                        b+=a;
                }
                SetDlgItemText(IDC_EDIT_RECMSG,b);
        }
}
这一段其实不难理解,因为前面两章已经介绍过多次了. sio_read是Pcomm的函数,从串口读取数据用. GetDlgItemText(IDC_EDIT_RECMSG,b);用于读出以前的历史记录,这样每次发上来的数据都放在后面连接起来.end是返回的收到的数据个数.用十六进制形式显示出来.


点击看大图
图片11 (原文件名:011.jpg) 

 
做完了上面一些,我们差一步就可以收到数据了.因为数据发上来后,底层响应,并调用了回调函数,在回调函数里面,发出一个消息WM_PCOMM 虽然我们在后面定义了一个Pcomm()函数专门用于响应这个消息,但这个自定义消息并不是自动连接到Pcomm()的,需要添加一个消息影射才能使WM_PCOMM消息影射到Pcomm()函数.双击左边[工作空间](WorkSpace)中的
DoDataExchange(CDataExchange* pDX) 
DoDataExchange这个函数的下面一般都这是用于定义消息影射的,将下面一段程序增加一行,完成后如下.
BEGIN_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_MAP
        ON_MESSAGE(WM_PCOMM,OnPcomm) //这里是消息影射
END_MESSAGE_MAP()


点击看大图
图片12 (原文件名:012.jpg) 

 
完成以后就可以按F7编译一下,如果无误就可以接收到数据了.运行后,选择正确的串口号,按一下[打开]按钮就可以了.
现在我们再来看看怎么发送数据
回到控件编辑状态,双击[发送]按钮,为该按钮添加代码.
void CEg03Dlg::OnBtnSend() 
{
        // TODO: Add your control notification handler code here
        CString a;
        unsigned char b=0;
        GetDlgItemText(IDC_EDIT_SEDMSG,a); //取得编辑框内所有文本
        a.MakeUpper();//全部转换为大写
        for(unsigned char i=0;i<a.GetLength();i+=3)
        {
                if(a.GetAt(i)>='A' && a.GetAt(i)<='Z') b=(a.GetAt(i)-55)*16; //判断填入的是字母还是数字,并把字符转换成十六进制数
                else if(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);
                else if(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用于发送一个字符.


ourdev_433349.jpg
图片13 (原文件名:013.jpg) 

 
填写待发送数据的时候要注意,每两位中间空格一下.填入的是十六进制数据.
下面再来总结一下静态方式调用动态库的方法.
1.        拷贝Lib,H头文件到工程路径
2.        在工程->设置中加入Lib模块.
3.        加入.h头文件,用于函数声明
4.        将Dll文件拷入到工程目标路径中
总结一下自定义消息方法:
1.        用#define WM_NAME WM_USER+1 定义一个自定义消息,名称随便.一般用WM开头.WM_USER+1中的1那个数字是自己定的,一个消息就无所谓了,喜欢多少都行,如果要定义很多个消息,不要冲突就行了
2.        在类定义里面声明一个消息响应函数,写成 afx_msg void FunctionName();格式.
3.        添加一个消息影射ON_MESSAGE(WM_NAME, FunctionName)注意这句后面是没有分号的.
4.        写好FunctionName的函数.
很简单的四步就行了.
这里是整个工程及教程的下载
工程文件ourdev_433389.rar(文件大小:1.72M) (原文件名:VC3.rar) 
教程第二章链接http://ouravr.com/bbs/bbs_content.jsp?bbs_sn=1481238&bbs_page_no=2&bbs_id=1000
教程第一章链接http://www.ouravr.com/bbs/bbs_content.jsp?bbs_sn=1455135&bbs_page_no=1&bbs_id=1000 

PARTNER CONTENT

文章评论2条评论)

登录后参与讨论

用户461316 2008-10-9 13:12

写的实在太好了

用户111934 2008-9-28 10:00

写的实在太好了
相关推荐阅读
jizzll_617398179 2010-02-08 10:53
正确理解A/D转换器的输入
http://www.freescale.com.cn/tech_ariticles/2005/1102_1.asp 许多嵌入式应用都会用到A/D转换器。然而,如果错误连接了A/D转换器输入端的电路,...
jizzll_617398179 2010-01-12 13:37
好久没来,都长草了
好久没来,都长草了,呵呵。整天忙得很,不过估计再过两个月会轻松点~\(≧▽≦)/~啦啦啦...
jizzll_617398179 2009-03-16 21:57
很奇怪,我收到站内信,但是却找不到发信人呢
发信人是个[],点击就跳到我自己的主页了。只好在这里给那位朋友留言了。 我不知道你说的哪篇文章,还有我也不知道你的EDN的ID。你可以在博客上面留言的。...
jizzll_617398179 2009-03-14 13:08
如何在EXCEL中使用16进制数
最近需要计算串口发送的数据,太多而且比较麻烦。知道EXECL功能强大,所以网上找了找,刚好,不错,很方便。网上找到的http://hi.bccn.net/space-99452-do-blog-id-...
jizzll_617398179 2009-03-05 14:37
〖常识〗不同晶振的最大波特率及其误差
最近犯了个错误,呵呵,晶振和波特率的问题,特查了记在这里,O(∩_∩)O~〖常识〗不同晶振的最大波特率及其误差http://www.aoxue.org/bbs/read.php?tid=65585不同...
jizzll_617398179 2009-02-11 14:30
大端模式和小端模式
转载http://www.cnblogs.com/TsuiLei/archive/2008/10/29/1322504.html大端格式:在这种格式中,字数据的高字节存储在低地址中,而字数据的低字节则...
EE直播间
更多
我要评论
2
10
关闭 站长推荐上一条 /3 下一条