VC++串口上位机简单例程源码及详细步骤.docx
《VC++串口上位机简单例程源码及详细步骤.docx》由会员分享,可在线阅读,更多相关《VC++串口上位机简单例程源码及详细步骤.docx(24页珍藏版)》请在冰豆网上搜索。
VC++串口上位机简单例程源码及详细步骤
VC++串口上位机简单例程(源码及详细步骤)
VC++串口上位机简单例程.rar(4.33MB)
下载次数:
156
2010-4-1315:
26
VC++编写简单串口上位机程序
2010年4月13日10:
23:
40
串口通信,MCU跟PC通信经常用到的一种通信方式,做界面、写上位机程序的编程语言、编译环境等不少,VB、C#、LABVIEW等等,我会的语言很少,C语言用得比较多,但是还没有找到如何用C语言来写串口通信上位机程序的资料,在图书管理找到了用VC++编写串口上位机的资料,参考书籍,用自己相当蹩脚的C++写出了一个简单的串口上位机程序,分享一下,体验一下单片机和PC通信的乐趣。
编译环境:
VC++6.0
操作系统:
VMWare虚拟出来的WindowsXP
程序实现功能:
1、PC初始化COM1口,使用n81方式,波特率57600与单片机通信。
PC的COM口编号可以通过如下方式修改:
1.jpg(58.4KB)
当然也可以通过上位机软件编写,通过按钮来选择COM端口号,但是此次仅仅是简单的例程,就没有弄那么复杂了。
COM1口可用的话,会提示串口初始化完毕。
否则会提示串口已经打开Portalreadyopen,表示串口已经打开,被占用了。
2、点击开始转换,串口会向单片机发送0xaa,单片机串口中断接收到0xaa后启动ADC转换一次,并把转换结果ADCL、ADCH共两个字节的结果发送至PC,PC进行数值转换后在窗口里显示。
(见文章末尾图)
3、为防止串口被一只占用,点击关闭串口可以关闭COM1,供其它程序使用,点击后按钮变为打开串口,点击可重新打开COM1。
程序的编写:
1、打开VC++6.0建立基于对话框的MFC应用程序Test,
2.jpg(47.33KB)
3.jpg(50.41KB)
4.jpg(57.05KB)
5.jpg(55.48KB)
6.jpg(62.67KB)
2、在项目中插入MSComm控件:
工程->增加到工程->ComponentsandControls->双击RegisteredActiveXControls->选择MicrosoftCommunicationsControl,version6.0->Insert,按默认值添加,你会发现多了个电话图标,这是增加后串口通信控件。
7.jpg(35.7KB)
8.jpg(55.81KB)
9.jpg(25.93KB)
3、删除确认、取消和提示框,添加“电话”、进程、静态文本、按钮、编辑框,拖动添加的控件,根据喜好布局。
10.jpg(42.66KB)
4、右击编辑框Edit选择属性,在样式里设置,勾选多行、垂直滚动,其它可按默认值。
右击静态文本Text选择属性,在常规设置里,修改标题。
右击按钮PushButton选择属性,在在常规设置里,修改标题。
修改后界面如下,程序写出来运行时“电话”标志会自动消失。
11.jpg(42.31KB)
5、查看->建立类向导MFCClassWizard->MemberViariable,选择ClassName为CTestDlg的类,ControlID为MSCOMM1,双击它,为它添加控制变量m_comm1。
类似的,选择IDC_BUTTON2添加控制变量m_serial。
(建立类向导也可以右击然后在弹出的快捷菜单里选择建立类向导)
12.jpg(50.11KB)
13.jpg(33.66KB)
至此,基本框架已经出来了,编译后运行可以看到如下所示的界面。
(组建->全部组件,然后组建->执行)
14.jpg(29.88KB)
6、点击左侧的视图窗口,可以在三种模式下切换,第三个是打开我们的源代码窗口,第一个是类,第二个是窗体的资源视图。
15.jpg(9.12KB)
选择FileView,展开testfiles->HeaderFiles,打开testDlg.h,在全局变量下添加如下代码,然后保存:
intgllen;//定义整型标量gllen,用于记录接收数据的个数
CProgressCtrl*pbar;//指向进度条的指针,用于操作进度条
CStringstrRXDdata;//编辑框显示的文本,记录历次转换值
16.jpg(37.54KB)
7、点击RecourseView,展开testrecourses->Dialog,双击IDD_TEST_DIALOG,编辑我们的主界面对话框。
双击击“电话”,弹出如下对话框,按确认键:
17.jpg(21.86KB)
VC会进入源码编辑窗口,这个函数是用来处理串口事件的,当PC串口接收到数据时,会产生一个数据缓冲区有数据的消息事件,然后调用执行这个函数。
添加如下代码,进行数据处理,窗口更新等操作:
VARIANTvariant1;//定义VARIANT型变量,用于存放接收到的数据
COleSafeArraysafearray;//定义safearray型变量
LONGlen,k;//定义长整型变量len,k
BYTErxdata[2048];//定义BYTE型数组
CStringstremp1,stremp2;//定义两个字符串
if(m_comm1.GetCommEvent()==2)//判断引起OnComm时间的原因
{//如果是接收到特定个字节数,则读取接收到的数据
variant1=m_comm1.GetInput();//把接收到的数据存放到VARIANT型变量里
safearray=variant1;//VARIANT型变量转换为ColeSafeArray型变量
len=safearray.GetOneDimSize();
for(k=0;k{safearray.GetElement(&k,rxdata+k);//得到接接收到的数据放到BYTE型数组rxdata里}for(k=0;k{BYTEbt=(*(unsignedchar*)(rxdata+k));//读取AD转换的高字节if((k%2)==0)if((k+1){gllen++;//全局的变量,对接收到的转换结果的个数进行计算stremp2.Format("第%d次转换结果:",gllen);//显示第几次转换inttemp=bt*4+((*(unsignedchar*)(rxdata+k+1))>>6);//高低字节合并成实际的转换结果,注意转换结果是左对齐 stremp1.Format("%2.2f",(2.56*temp/1024));//计算成实际电压值SetDlgItemText(IDC_STATIC,("当前电压值为:"+stremp1+"V"));//更新静态文本控件pbar->SetPos(temp);//更新进度条的当前位置strRXDdata+=stremp2;//把新的数据放到全局的字符串里 strRXDdata+=stremp1;strRXDdata+="V\r\n";//字符串加单位V后换行}}}SetDlgItemText(IDC_EDIT1,strRXDdata);//更新文本控件的显示18.jpg(44.5KB)2010-4-1315:26这时重新编译一下,看会不会有什么错误,出现下面提示,可以选择全部组建来清除。LINK:LNK4073:cannotcreatemapfor.ILKfile;linkingnonincrementally出现下面错误,请关闭运行的test.exe后重试。LINK:fatalerrorLNK1104:cannotopenfile"Debug/test.exe"出现下面错误两种错误,是由于空间编号问题引起的,当我们添加了编辑框或者“电话”后再添加,其编号自动加一,就会出现控件没定义。Z:\vc++串口上位机\test\testDlg.cpp(32):errorC2065:'IDC_MSCOMM1':undeclaredidentifierZ:\vc++串口上位机\test\testDlg.cpp(139):errorC2065:'IDC_EDIT1':undeclaredidentifier解决方法是,在RecourseView里,打开窗体IDD_TEST_DIALOG,右击“电话”或者编辑框等其它出错的控件,右击选择属性,在常规里修改ID,这里的程序,除BUTTON有1、2两个之外,其它都是119.jpg(16.02KB)2010-4-1315:26全部组建编译一下,看看有没有错误,没有错误就可以运行一下,可以看到界面更原来是一样的。有错误就修改一下,省得弄多了,错在哪里都不知道,查起来麻烦。 8、在源码编辑里,打开testDlg.cpp文件,进行窗口初始化函数的编写。找到BOOLCTestDlg::OnInitDialog()函数,在SetIcon(m_hIcon,FALSE); //Setsmallicon//TODO:Addextrainitializationhere后面添加如下初始化代码:gllen=0;//记录转换次数全局变量清零if(!m_comm1.GetPortOpen())//判断串口是否已经打开{m_comm1.SetCommPort(1);//选择串口号1m_comm1.SetPortOpen(TRUE);//打开串口m_comm1.SetRThreshold(2);//收到两个字节引发OnComm事件m_comm1.SetInputMode(1);//输入模式选为二进制m_comm1.SetSettings("57600,n,8,1");//设置串口参数,波特率57600,无奇偶校验,1位停止位,8位数据位MessageBox("串口初始化完毕","提示");//提示串口成功初始化}elseMessageBox("串口被占用","提示");//如果已经打开串口,消息框提醒pbar=(CProgressCtrl*)GetDlgItem(IDC_PROGRESS1);//获得指向IDC_PROGRESS1的指针pbar->SetRange(0,1023);//设置进度条的范围0~1023pbar->SetPos(0);//当前位置为0m_serial.SetWindowText("关闭串口");//按钮显示状态改变20.jpg(47.05KB)2010-4-1315:26可以看到,串口的参数等等都在在这里初始化的,可以根据自己的需要修改的,具体可以查看VC++里的详细介绍,看看有哪些参数可以给我们修改来用。添加后再编译一下,运行后可以看到多了一个串口初始化的提示信息窗口。至此,我们已经完成了主要的串口操作及界面,剩下的就是两个按钮的操作了。 9、回到资源视图的IDD_TEST_DIALOG窗口,双击开始转换按钮,给它添加事件,点击后PC通过串口发送0xaa出来,给单片机接收。添加如下代码:CByteArraym_Array;//定义字节数组m_Array.RemoveAll();//字节数组清空m_Array.SetSize(1);//设定维数为1m_Array.SetAt(0,0xaa);//给m_array[0]赋值0m_comm1.SetOutput(COleVariant(m_Array));//由于SetOutput函数的参数为VARIANT型,必须强制转换后才能发送21.jpg(31.37KB)2010-4-1315:26同样地,双击另外一个按钮,给串口操作按钮添加代码,用于关闭或者打开串口。添加如下代码:if(!m_comm1.GetPortOpen())//判断串口是否已经打开{ m_comm1.SetPortOpen(TRUE);//如果串口是关闭的,则打开串口m_serial.SetWindowText("关闭串口");//按钮显示状态改变}else{m_comm1.SetPortOpen(FALSE);//如果已经打开串口,则关闭串口m_serial.SetWindowText("打开串口");//按钮显示状态改变}22.jpg(43.02KB)2010-4-1315:26至此,一个简单的串口上位机软件编写完成了,可以用来测试下,通过单片机往串口里发送数据,可以看到主窗口的的转换结果,已经进度条显示电压值变化。要把这个程序拿出来用,只需把…\vc++串口上位机\test\Release的test.exe拷出来用就行。Release可以在编译窗口里选择win32release,然后重新编译一下就出来了。23.jpg(23.28KB)2010-4-1315:26 1。建立mfc工程,都会撒。 将控件加进来:打开“Project->AddToProject->ComponentsandControls->RegisteredActivexControls”,然后选择控件:MicrosoftCommunicationControl,version6.0插入到当前的工程中。这样就将类CMSComm的相关文件mscomm.cpp和mscomm.h一并加入到了工程中。编程时只需将控件对话中的MSComm控件拖至你的应用对话框中就OK了2。定义串口对象: CMSComm m_MSComm;3。串口初始化: DWORDstyle=WS_VISIBLE;m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM);if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口{m_MSComm.SetPortOpen(FALSE);}m_MSComm.SetCommPort(1);//选择COM1m_MSComm.SetInBufferSize(1024);//接收缓冲区m_MSComm.SetOutBufferSize(1024);//发送缓冲区m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取m_MSComm.SetInputMode(1);//以二进制方式读写数据m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开m_MSComm.SetPortOpen(TRUE);//打开串口else{m_MSComm.SetOutBufferCount(0);AfxMessageBox("OpenTheSerialPort1Failurre!");}4。串口数据读写: MSComm类的读写函数比较简单:GetInput()和SetOutput()。函数原形分别为VARIANTGetInput()和voidSetOutput(constVARIANTnewValue),均使用VARIANT类型。但PC机发送和接收数据时习惯用字符串形式。MSDN中查阅VARIANT类型,可以用BSTR表示字符串,但所有的BSTR都包含宽字符,而只有WindowsNT支持宽字符,Windows9X并不支持。所以要完成一个适应各平台的串口应用程序必须解决这个问题。这里使用CbyteArray即可解决之。发数据:在对话框对加入按钮控件并给你添加消息 voidCTest_mscommDlg::OnSend(){//TODO:Addyourcontrolnotificationhandlercodehereinti,Count;CStringm_SendData;m_SendData="Hello!";Count=m_SendData.GetLength();CByteArraym_Array;m_Array.RemoveAll();m_Array.SetSize(Count);for(inti=0;i{ m_Array.SetAt(i,m_SendData.GetAt(i));}m_mscomm.SetOutput(COleVariant(m_Array);}收数据:给串口控件添加消息voidCTest_mscommDlg::OnOnCommMscomm(){VARIANTm_input;char*str,*str1;intk,nEvent,i;CStringstr2,m_RcvData;nEvent=m_MSComm.GetCommEvent();switch(nEvent){case2: k=m_MSComm.GetInBufferCount(); //接收缓冲区的字符数目 if(k>0) { m_input=m_MSComm.GetInput(); str=(char*)(unsignedchar*)m_input.parray->pvData; } i=0; str1=str; while(im_disp+=m_RcvData;UpdateData(false);}首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project-->AddtoProject-->ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。此时,你只需要关心控件提供的对Windows通讯驱动程序的API函数的接口。换句话说,只需要设置和监视MSComm控件的属性和事件。打开所需串口后,需要考虑串口通信的时机。在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应//若是在SDI中使用该控件则要调用下两句,在对话框程序中该语句有MFC自己创建 //所以不用人为添加 DWORDstyle=WS_VISIBLE; VC++下用MSComm控件实现串口通讯·2005-10-0817:59:47·来源:天极网m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); //串口控件的初始化 DWORDstyle=WS_VISIBLE; m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口 { m_MSComm.SetPortOpen(FALSE); } m_MSComm.SetCommPort(1);//选择COM1 m_MSComm.SetInBufferSize(1024);//接收缓冲区 m_MSComm.SetOutBufferSize(1024);//发送缓冲区 m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取 m_MSComm.SetInputMode(1);//以二进制方式读写数据 m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件 m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位 if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开 m_MSComm.SetPortOpen(TRUE);//打开串口 else m_MSComm.SetOutBufferCount(0); //控件事件的响应声明 //*.h //{{AFX_MSG(CGolfView) afx_msgBOOLOnComm(); DECLARE_EVENTSINK_MAP() //}}AFX_MSG //*.cpp BEGIN_EVENTSINK_MAP(CGolfView,CView) //{{AFX_EVENTSINK_MAP(CAboutDlg) ON_EVENT(CGolfView,IDC_MSCOMM1,1/*OnComm*/,OnComm,VTS_NONE) //}}AFX_EVENTSINK_MAP END_EVENTSINK_MAP() //控件事件的响应 BOOLCGolfView::OnComm() { VARIANTvariant_inp; COleSafeArraysafearray_inp; LONGlen,k; BYTErxdata[2
{
safearray.GetElement(&k,rxdata+k);//得到接接收到的数据放到BYTE型数组rxdata里
}
for(k=0;k{BYTEbt=(*(unsignedchar*)(rxdata+k));//读取AD转换的高字节if((k%2)==0)if((k+1){gllen++;//全局的变量,对接收到的转换结果的个数进行计算stremp2.Format("第%d次转换结果:",gllen);//显示第几次转换inttemp=bt*4+((*(unsignedchar*)(rxdata+k+1))>>6);//高低字节合并成实际的转换结果,注意转换结果是左对齐 stremp1.Format("%2.2f",(2.56*temp/1024));//计算成实际电压值SetDlgItemText(IDC_STATIC,("当前电压值为:"+stremp1+"V"));//更新静态文本控件pbar->SetPos(temp);//更新进度条的当前位置strRXDdata+=stremp2;//把新的数据放到全局的字符串里 strRXDdata+=stremp1;strRXDdata+="V\r\n";//字符串加单位V后换行}}}SetDlgItemText(IDC_EDIT1,strRXDdata);//更新文本控件的显示18.jpg(44.5KB)2010-4-1315:26这时重新编译一下,看会不会有什么错误,出现下面提示,可以选择全部组建来清除。LINK:LNK4073:cannotcreatemapfor.ILKfile;linkingnonincrementally出现下面错误,请关闭运行的test.exe后重试。LINK:fatalerrorLNK1104:cannotopenfile"Debug/test.exe"出现下面错误两种错误,是由于空间编号问题引起的,当我们添加了编辑框或者“电话”后再添加,其编号自动加一,就会出现控件没定义。Z:\vc++串口上位机\test\testDlg.cpp(32):errorC2065:'IDC_MSCOMM1':undeclaredidentifierZ:\vc++串口上位机\test\testDlg.cpp(139):errorC2065:'IDC_EDIT1':undeclaredidentifier解决方法是,在RecourseView里,打开窗体IDD_TEST_DIALOG,右击“电话”或者编辑框等其它出错的控件,右击选择属性,在常规里修改ID,这里的程序,除BUTTON有1、2两个之外,其它都是119.jpg(16.02KB)2010-4-1315:26全部组建编译一下,看看有没有错误,没有错误就可以运行一下,可以看到界面更原来是一样的。有错误就修改一下,省得弄多了,错在哪里都不知道,查起来麻烦。 8、在源码编辑里,打开testDlg.cpp文件,进行窗口初始化函数的编写。找到BOOLCTestDlg::OnInitDialog()函数,在SetIcon(m_hIcon,FALSE); //Setsmallicon//TODO:Addextrainitializationhere后面添加如下初始化代码:gllen=0;//记录转换次数全局变量清零if(!m_comm1.GetPortOpen())//判断串口是否已经打开{m_comm1.SetCommPort(1);//选择串口号1m_comm1.SetPortOpen(TRUE);//打开串口m_comm1.SetRThreshold(2);//收到两个字节引发OnComm事件m_comm1.SetInputMode(1);//输入模式选为二进制m_comm1.SetSettings("57600,n,8,1");//设置串口参数,波特率57600,无奇偶校验,1位停止位,8位数据位MessageBox("串口初始化完毕","提示");//提示串口成功初始化}elseMessageBox("串口被占用","提示");//如果已经打开串口,消息框提醒pbar=(CProgressCtrl*)GetDlgItem(IDC_PROGRESS1);//获得指向IDC_PROGRESS1的指针pbar->SetRange(0,1023);//设置进度条的范围0~1023pbar->SetPos(0);//当前位置为0m_serial.SetWindowText("关闭串口");//按钮显示状态改变20.jpg(47.05KB)2010-4-1315:26可以看到,串口的参数等等都在在这里初始化的,可以根据自己的需要修改的,具体可以查看VC++里的详细介绍,看看有哪些参数可以给我们修改来用。添加后再编译一下,运行后可以看到多了一个串口初始化的提示信息窗口。至此,我们已经完成了主要的串口操作及界面,剩下的就是两个按钮的操作了。 9、回到资源视图的IDD_TEST_DIALOG窗口,双击开始转换按钮,给它添加事件,点击后PC通过串口发送0xaa出来,给单片机接收。添加如下代码:CByteArraym_Array;//定义字节数组m_Array.RemoveAll();//字节数组清空m_Array.SetSize(1);//设定维数为1m_Array.SetAt(0,0xaa);//给m_array[0]赋值0m_comm1.SetOutput(COleVariant(m_Array));//由于SetOutput函数的参数为VARIANT型,必须强制转换后才能发送21.jpg(31.37KB)2010-4-1315:26同样地,双击另外一个按钮,给串口操作按钮添加代码,用于关闭或者打开串口。添加如下代码:if(!m_comm1.GetPortOpen())//判断串口是否已经打开{ m_comm1.SetPortOpen(TRUE);//如果串口是关闭的,则打开串口m_serial.SetWindowText("关闭串口");//按钮显示状态改变}else{m_comm1.SetPortOpen(FALSE);//如果已经打开串口,则关闭串口m_serial.SetWindowText("打开串口");//按钮显示状态改变}22.jpg(43.02KB)2010-4-1315:26至此,一个简单的串口上位机软件编写完成了,可以用来测试下,通过单片机往串口里发送数据,可以看到主窗口的的转换结果,已经进度条显示电压值变化。要把这个程序拿出来用,只需把…\vc++串口上位机\test\Release的test.exe拷出来用就行。Release可以在编译窗口里选择win32release,然后重新编译一下就出来了。23.jpg(23.28KB)2010-4-1315:26 1。建立mfc工程,都会撒。 将控件加进来:打开“Project->AddToProject->ComponentsandControls->RegisteredActivexControls”,然后选择控件:MicrosoftCommunicationControl,version6.0插入到当前的工程中。这样就将类CMSComm的相关文件mscomm.cpp和mscomm.h一并加入到了工程中。编程时只需将控件对话中的MSComm控件拖至你的应用对话框中就OK了2。定义串口对象: CMSComm m_MSComm;3。串口初始化: DWORDstyle=WS_VISIBLE;m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM);if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口{m_MSComm.SetPortOpen(FALSE);}m_MSComm.SetCommPort(1);//选择COM1m_MSComm.SetInBufferSize(1024);//接收缓冲区m_MSComm.SetOutBufferSize(1024);//发送缓冲区m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取m_MSComm.SetInputMode(1);//以二进制方式读写数据m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开m_MSComm.SetPortOpen(TRUE);//打开串口else{m_MSComm.SetOutBufferCount(0);AfxMessageBox("OpenTheSerialPort1Failurre!");}4。串口数据读写: MSComm类的读写函数比较简单:GetInput()和SetOutput()。函数原形分别为VARIANTGetInput()和voidSetOutput(constVARIANTnewValue),均使用VARIANT类型。但PC机发送和接收数据时习惯用字符串形式。MSDN中查阅VARIANT类型,可以用BSTR表示字符串,但所有的BSTR都包含宽字符,而只有WindowsNT支持宽字符,Windows9X并不支持。所以要完成一个适应各平台的串口应用程序必须解决这个问题。这里使用CbyteArray即可解决之。发数据:在对话框对加入按钮控件并给你添加消息 voidCTest_mscommDlg::OnSend(){//TODO:Addyourcontrolnotificationhandlercodehereinti,Count;CStringm_SendData;m_SendData="Hello!";Count=m_SendData.GetLength();CByteArraym_Array;m_Array.RemoveAll();m_Array.SetSize(Count);for(inti=0;i{ m_Array.SetAt(i,m_SendData.GetAt(i));}m_mscomm.SetOutput(COleVariant(m_Array);}收数据:给串口控件添加消息voidCTest_mscommDlg::OnOnCommMscomm(){VARIANTm_input;char*str,*str1;intk,nEvent,i;CStringstr2,m_RcvData;nEvent=m_MSComm.GetCommEvent();switch(nEvent){case2: k=m_MSComm.GetInBufferCount(); //接收缓冲区的字符数目 if(k>0) { m_input=m_MSComm.GetInput(); str=(char*)(unsignedchar*)m_input.parray->pvData; } i=0; str1=str; while(im_disp+=m_RcvData;UpdateData(false);}首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project-->AddtoProject-->ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。此时,你只需要关心控件提供的对Windows通讯驱动程序的API函数的接口。换句话说,只需要设置和监视MSComm控件的属性和事件。打开所需串口后,需要考虑串口通信的时机。在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应//若是在SDI中使用该控件则要调用下两句,在对话框程序中该语句有MFC自己创建 //所以不用人为添加 DWORDstyle=WS_VISIBLE; VC++下用MSComm控件实现串口通讯·2005-10-0817:59:47·来源:天极网m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); //串口控件的初始化 DWORDstyle=WS_VISIBLE; m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口 { m_MSComm.SetPortOpen(FALSE); } m_MSComm.SetCommPort(1);//选择COM1 m_MSComm.SetInBufferSize(1024);//接收缓冲区 m_MSComm.SetOutBufferSize(1024);//发送缓冲区 m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取 m_MSComm.SetInputMode(1);//以二进制方式读写数据 m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件 m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位 if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开 m_MSComm.SetPortOpen(TRUE);//打开串口 else m_MSComm.SetOutBufferCount(0); //控件事件的响应声明 //*.h //{{AFX_MSG(CGolfView) afx_msgBOOLOnComm(); DECLARE_EVENTSINK_MAP() //}}AFX_MSG //*.cpp BEGIN_EVENTSINK_MAP(CGolfView,CView) //{{AFX_EVENTSINK_MAP(CAboutDlg) ON_EVENT(CGolfView,IDC_MSCOMM1,1/*OnComm*/,OnComm,VTS_NONE) //}}AFX_EVENTSINK_MAP END_EVENTSINK_MAP() //控件事件的响应 BOOLCGolfView::OnComm() { VARIANTvariant_inp; COleSafeArraysafearray_inp; LONGlen,k; BYTErxdata[2
BYTEbt=(*(unsignedchar*)(rxdata+k));//读取AD转换的高字节
if((k%2)==0)
if((k+1){gllen++;//全局的变量,对接收到的转换结果的个数进行计算stremp2.Format("第%d次转换结果:",gllen);//显示第几次转换inttemp=bt*4+((*(unsignedchar*)(rxdata+k+1))>>6);//高低字节合并成实际的转换结果,注意转换结果是左对齐 stremp1.Format("%2.2f",(2.56*temp/1024));//计算成实际电压值SetDlgItemText(IDC_STATIC,("当前电压值为:"+stremp1+"V"));//更新静态文本控件pbar->SetPos(temp);//更新进度条的当前位置strRXDdata+=stremp2;//把新的数据放到全局的字符串里 strRXDdata+=stremp1;strRXDdata+="V\r\n";//字符串加单位V后换行}}}SetDlgItemText(IDC_EDIT1,strRXDdata);//更新文本控件的显示18.jpg(44.5KB)2010-4-1315:26这时重新编译一下,看会不会有什么错误,出现下面提示,可以选择全部组建来清除。LINK:LNK4073:cannotcreatemapfor.ILKfile;linkingnonincrementally出现下面错误,请关闭运行的test.exe后重试。LINK:fatalerrorLNK1104:cannotopenfile"Debug/test.exe"出现下面错误两种错误,是由于空间编号问题引起的,当我们添加了编辑框或者“电话”后再添加,其编号自动加一,就会出现控件没定义。Z:\vc++串口上位机\test\testDlg.cpp(32):errorC2065:'IDC_MSCOMM1':undeclaredidentifierZ:\vc++串口上位机\test\testDlg.cpp(139):errorC2065:'IDC_EDIT1':undeclaredidentifier解决方法是,在RecourseView里,打开窗体IDD_TEST_DIALOG,右击“电话”或者编辑框等其它出错的控件,右击选择属性,在常规里修改ID,这里的程序,除BUTTON有1、2两个之外,其它都是119.jpg(16.02KB)2010-4-1315:26全部组建编译一下,看看有没有错误,没有错误就可以运行一下,可以看到界面更原来是一样的。有错误就修改一下,省得弄多了,错在哪里都不知道,查起来麻烦。 8、在源码编辑里,打开testDlg.cpp文件,进行窗口初始化函数的编写。找到BOOLCTestDlg::OnInitDialog()函数,在SetIcon(m_hIcon,FALSE); //Setsmallicon//TODO:Addextrainitializationhere后面添加如下初始化代码:gllen=0;//记录转换次数全局变量清零if(!m_comm1.GetPortOpen())//判断串口是否已经打开{m_comm1.SetCommPort(1);//选择串口号1m_comm1.SetPortOpen(TRUE);//打开串口m_comm1.SetRThreshold(2);//收到两个字节引发OnComm事件m_comm1.SetInputMode(1);//输入模式选为二进制m_comm1.SetSettings("57600,n,8,1");//设置串口参数,波特率57600,无奇偶校验,1位停止位,8位数据位MessageBox("串口初始化完毕","提示");//提示串口成功初始化}elseMessageBox("串口被占用","提示");//如果已经打开串口,消息框提醒pbar=(CProgressCtrl*)GetDlgItem(IDC_PROGRESS1);//获得指向IDC_PROGRESS1的指针pbar->SetRange(0,1023);//设置进度条的范围0~1023pbar->SetPos(0);//当前位置为0m_serial.SetWindowText("关闭串口");//按钮显示状态改变20.jpg(47.05KB)2010-4-1315:26可以看到,串口的参数等等都在在这里初始化的,可以根据自己的需要修改的,具体可以查看VC++里的详细介绍,看看有哪些参数可以给我们修改来用。添加后再编译一下,运行后可以看到多了一个串口初始化的提示信息窗口。至此,我们已经完成了主要的串口操作及界面,剩下的就是两个按钮的操作了。 9、回到资源视图的IDD_TEST_DIALOG窗口,双击开始转换按钮,给它添加事件,点击后PC通过串口发送0xaa出来,给单片机接收。添加如下代码:CByteArraym_Array;//定义字节数组m_Array.RemoveAll();//字节数组清空m_Array.SetSize(1);//设定维数为1m_Array.SetAt(0,0xaa);//给m_array[0]赋值0m_comm1.SetOutput(COleVariant(m_Array));//由于SetOutput函数的参数为VARIANT型,必须强制转换后才能发送21.jpg(31.37KB)2010-4-1315:26同样地,双击另外一个按钮,给串口操作按钮添加代码,用于关闭或者打开串口。添加如下代码:if(!m_comm1.GetPortOpen())//判断串口是否已经打开{ m_comm1.SetPortOpen(TRUE);//如果串口是关闭的,则打开串口m_serial.SetWindowText("关闭串口");//按钮显示状态改变}else{m_comm1.SetPortOpen(FALSE);//如果已经打开串口,则关闭串口m_serial.SetWindowText("打开串口");//按钮显示状态改变}22.jpg(43.02KB)2010-4-1315:26至此,一个简单的串口上位机软件编写完成了,可以用来测试下,通过单片机往串口里发送数据,可以看到主窗口的的转换结果,已经进度条显示电压值变化。要把这个程序拿出来用,只需把…\vc++串口上位机\test\Release的test.exe拷出来用就行。Release可以在编译窗口里选择win32release,然后重新编译一下就出来了。23.jpg(23.28KB)2010-4-1315:26 1。建立mfc工程,都会撒。 将控件加进来:打开“Project->AddToProject->ComponentsandControls->RegisteredActivexControls”,然后选择控件:MicrosoftCommunicationControl,version6.0插入到当前的工程中。这样就将类CMSComm的相关文件mscomm.cpp和mscomm.h一并加入到了工程中。编程时只需将控件对话中的MSComm控件拖至你的应用对话框中就OK了2。定义串口对象: CMSComm m_MSComm;3。串口初始化: DWORDstyle=WS_VISIBLE;m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM);if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口{m_MSComm.SetPortOpen(FALSE);}m_MSComm.SetCommPort(1);//选择COM1m_MSComm.SetInBufferSize(1024);//接收缓冲区m_MSComm.SetOutBufferSize(1024);//发送缓冲区m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取m_MSComm.SetInputMode(1);//以二进制方式读写数据m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开m_MSComm.SetPortOpen(TRUE);//打开串口else{m_MSComm.SetOutBufferCount(0);AfxMessageBox("OpenTheSerialPort1Failurre!");}4。串口数据读写: MSComm类的读写函数比较简单:GetInput()和SetOutput()。函数原形分别为VARIANTGetInput()和voidSetOutput(constVARIANTnewValue),均使用VARIANT类型。但PC机发送和接收数据时习惯用字符串形式。MSDN中查阅VARIANT类型,可以用BSTR表示字符串,但所有的BSTR都包含宽字符,而只有WindowsNT支持宽字符,Windows9X并不支持。所以要完成一个适应各平台的串口应用程序必须解决这个问题。这里使用CbyteArray即可解决之。发数据:在对话框对加入按钮控件并给你添加消息 voidCTest_mscommDlg::OnSend(){//TODO:Addyourcontrolnotificationhandlercodehereinti,Count;CStringm_SendData;m_SendData="Hello!";Count=m_SendData.GetLength();CByteArraym_Array;m_Array.RemoveAll();m_Array.SetSize(Count);for(inti=0;i{ m_Array.SetAt(i,m_SendData.GetAt(i));}m_mscomm.SetOutput(COleVariant(m_Array);}收数据:给串口控件添加消息voidCTest_mscommDlg::OnOnCommMscomm(){VARIANTm_input;char*str,*str1;intk,nEvent,i;CStringstr2,m_RcvData;nEvent=m_MSComm.GetCommEvent();switch(nEvent){case2: k=m_MSComm.GetInBufferCount(); //接收缓冲区的字符数目 if(k>0) { m_input=m_MSComm.GetInput(); str=(char*)(unsignedchar*)m_input.parray->pvData; } i=0; str1=str; while(im_disp+=m_RcvData;UpdateData(false);}首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project-->AddtoProject-->ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。此时,你只需要关心控件提供的对Windows通讯驱动程序的API函数的接口。换句话说,只需要设置和监视MSComm控件的属性和事件。打开所需串口后,需要考虑串口通信的时机。在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应//若是在SDI中使用该控件则要调用下两句,在对话框程序中该语句有MFC自己创建 //所以不用人为添加 DWORDstyle=WS_VISIBLE; VC++下用MSComm控件实现串口通讯·2005-10-0817:59:47·来源:天极网m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); //串口控件的初始化 DWORDstyle=WS_VISIBLE; m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口 { m_MSComm.SetPortOpen(FALSE); } m_MSComm.SetCommPort(1);//选择COM1 m_MSComm.SetInBufferSize(1024);//接收缓冲区 m_MSComm.SetOutBufferSize(1024);//发送缓冲区 m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取 m_MSComm.SetInputMode(1);//以二进制方式读写数据 m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件 m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位 if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开 m_MSComm.SetPortOpen(TRUE);//打开串口 else m_MSComm.SetOutBufferCount(0); //控件事件的响应声明 //*.h //{{AFX_MSG(CGolfView) afx_msgBOOLOnComm(); DECLARE_EVENTSINK_MAP() //}}AFX_MSG //*.cpp BEGIN_EVENTSINK_MAP(CGolfView,CView) //{{AFX_EVENTSINK_MAP(CAboutDlg) ON_EVENT(CGolfView,IDC_MSCOMM1,1/*OnComm*/,OnComm,VTS_NONE) //}}AFX_EVENTSINK_MAP END_EVENTSINK_MAP() //控件事件的响应 BOOLCGolfView::OnComm() { VARIANTvariant_inp; COleSafeArraysafearray_inp; LONGlen,k; BYTErxdata[2
gllen++;//全局的变量,对接收到的转换结果的个数进行计算
stremp2.Format("第%d次转换结果:
",gllen);//显示第几次转换
inttemp=bt*4+((*(unsignedchar*)(rxdata+k+1))>>6);//高低字节合并成实际的转换结果,注意转换结果是左对齐
stremp1.Format("%2.2f",(2.56*temp/1024));//计算成实际电压值
SetDlgItemText(IDC_STATIC,("当前电压值为:
"+stremp1+"V"));//更新静态文本控件
pbar->SetPos(temp);//更新进度条的当前位置
strRXDdata+=stremp2;//把新的数据放到全局的字符串里
strRXDdata+=stremp1;
strRXDdata+="V\r\n";//字符串加单位V后换行
SetDlgItemText(IDC_EDIT1,strRXDdata);//更新文本控件的显示
18.jpg(44.5KB)
这时重新编译一下,看会不会有什么错误,出现下面提示,可以选择全部组建来清除。
LINK:
LNK4073:
cannotcreatemapfor.ILKfile;linkingnonincrementally
出现下面错误,请关闭运行的test.exe后重试。
fatalerrorLNK1104:
cannotopenfile"Debug/test.exe"
出现下面错误两种错误,是由于空间编号问题引起的,当我们添加了编辑框或者“电话”后再添加,其编号自动加一,就会出现控件没定义。
Z:
\vc++串口上位机\test\testDlg.cpp(32):
errorC2065:
'IDC_MSCOMM1':
undeclaredidentifier
\vc++串口上位机\test\testDlg.cpp(139):
'IDC_EDIT1':
解决方法是,在RecourseView里,打开窗体IDD_TEST_DIALOG,右击“电话”或者编辑框等其它出错的控件,右击选择属性,在常规里修改ID,这里的程序,除BUTTON有1、2两个之外,其它都是1
19.jpg(16.02KB)
全部组建编译一下,看看有没有错误,没有错误就可以运行一下,可以看到界面更原来是一样的。
有错误就修改一下,省得弄多了,错在哪里都不知道,查起来麻烦。
8、在源码编辑里,打开testDlg.cpp文件,进行窗口初始化函数的编写。
找到BOOLCTestDlg:
:
OnInitDialog()函数,
在SetIcon(m_hIcon,FALSE); //Setsmallicon
//TODO:
Addextrainitializationhere
后面添加如下初始化代码:
gllen=0;//记录转换次数全局变量清零
if(!
m_comm1.GetPortOpen())//判断串口是否已经打开
m_comm1.SetCommPort
(1);//选择串口号1
m_comm1.SetPortOpen(TRUE);//打开串口
m_comm1.SetRThreshold
(2);//收到两个字节引发OnComm事件
m_comm1.SetInputMode
(1);//输入模式选为二进制
m_comm1.SetSettings("57600,n,8,1");//设置串口参数,波特率57600,无奇偶校验,1位停止位,8位数据位
MessageBox("串口初始化完毕","提示");//提示串口成功初始化
elseMessageBox("串口被占用","提示");//如果已经打开串口,消息框提醒
pbar=(CProgressCtrl*)GetDlgItem(IDC_PROGRESS1);//获得指向IDC_PROGRESS1的指针
pbar->SetRange(0,1023);//设置进度条的范围0~1023
pbar->SetPos(0);//当前位置为0
m_serial.SetWindowText("关闭串口");//按钮显示状态改变
20.jpg(47.05KB)
可以看到,串口的参数等等都在在这里初始化的,可以根据自己的需要修改的,具体可以查看VC++里的详细介绍,看看有哪些参数可以给我们修改来用。
添加后再编译一下,运行后可以看到多了一个串口初始化的提示信息窗口。
至此,我们已经完成了主要的串口操作及界面,剩下的就是两个按钮的操作了。
9、回到资源视图的IDD_TEST_DIALOG窗口,双击开始转换按钮,给它添加事件,点击后PC通过串口发送0xaa出来,给单片机接收。
添加如下代码:
CByteArraym_Array;//定义字节数组
m_Array.RemoveAll();//字节数组清空
m_Array.SetSize
(1);//设定维数为1
m_Array.SetAt(0,0xaa);//给m_array[0]赋值0
m_comm1.SetOutput(COleVariant(m_Array));//由于SetOutput函数的参数为VARIANT型,必须强制转换后才能发送
21.jpg(31.37KB)
同样地,双击另外一个按钮,给串口操作按钮添加代码,用于关闭或者打开串口。
添加如下代码:
m_comm1.SetPortOpen(TRUE);//如果串口是关闭的,则打开串口
else
m_comm1.SetPortOpen(FALSE);//如果已经打开串口,则关闭串口
m_serial.SetWindowText("打开串口");//按钮显示状态改变
22.jpg(43.02KB)
至此,一个简单的串口上位机软件编写完成了,可以用来测试下,通过单片机往串口里发送数据,可以看到主窗口的的转换结果,已经进度条显示电压值变化。
要把这个程序拿出来用,只需把…\vc++串口上位机\test\Release的test.exe拷出来用就行。
Release可以在编译窗口里选择win32release,然后重新编译一下就出来了。
23.jpg(23.28KB)
1。
建立mfc工程,都会撒。
将控件加进来:
打开“Project->AddToProject->ComponentsandControls->RegisteredActivexControls”,然后选择控件:
MicrosoftCommunicationControl,version6.0插入到当前的工程中。
这样就将类CMSComm的相关文件mscomm.cpp和mscomm.h一并加入到了工程中。
编程时只需将控件对话中的MSComm控件拖至你的应用对话框中就OK了
2。
定义串口对象:
CMSComm m_MSComm;
3。
串口初始化:
DWORDstyle=WS_VISIBLE;
m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM);
if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口
m_MSComm.SetPortOpen(FALSE);
m_MSComm.SetCommPort
(1);//选择COM1
m_MSComm.SetInBufferSize(1024);//接收缓冲区
m_MSComm.SetOutBufferSize(1024);//发送缓冲区
m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取
m_MSComm.SetInputMode
(1);//以二进制方式读写数据
m_MSComm.SetRThreshold
(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件
m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位
m_MSComm.GetPortOpen())//如果串口没有打开则打开
m_MSComm.SetPortOpen(TRUE);//打开串口
m_MSComm.SetOutBufferCount(0);
AfxMessageBox("OpenTheSerialPort1Failurre!
");
4。
串口数据读写:
MSComm类的读写函数比较简单:
GetInput()和SetOutput()。
函数原形分别为VARIANTGetInput()和voidSetOutput(constVARIANTnewValue),均使用VARIANT类型。
但PC机发送和接收数据时习惯用字符串形式。
MSDN中查阅VARIANT类型,可以用BSTR表示字符串,但所有的BSTR都包含宽字符,而只有WindowsNT支持宽字符,Windows9X并不支持。
所以要完成一个适应各平台的串口应用程序必须解决这个问题。
这里使用CbyteArray即可解决之。
发数据:
在对话框对加入按钮控件并给你添加消息
voidCTest_mscommDlg:
OnSend()
Addyourcontrolnotificationhandlercodehere
inti,Count;
CStringm_SendData;
m_SendData="Hello!
";
Count=m_SendData.GetLength();
CByteArraym_Array;
m_Array.RemoveAll();
m_Array.SetSize(Count);
for(inti=0;i{ m_Array.SetAt(i,m_SendData.GetAt(i));}m_mscomm.SetOutput(COleVariant(m_Array);}收数据:给串口控件添加消息voidCTest_mscommDlg::OnOnCommMscomm(){VARIANTm_input;char*str,*str1;intk,nEvent,i;CStringstr2,m_RcvData;nEvent=m_MSComm.GetCommEvent();switch(nEvent){case2: k=m_MSComm.GetInBufferCount(); //接收缓冲区的字符数目 if(k>0) { m_input=m_MSComm.GetInput(); str=(char*)(unsignedchar*)m_input.parray->pvData; } i=0; str1=str; while(im_disp+=m_RcvData;UpdateData(false);}首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project-->AddtoProject-->ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。此时,你只需要关心控件提供的对Windows通讯驱动程序的API函数的接口。换句话说,只需要设置和监视MSComm控件的属性和事件。打开所需串口后,需要考虑串口通信的时机。在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应//若是在SDI中使用该控件则要调用下两句,在对话框程序中该语句有MFC自己创建 //所以不用人为添加 DWORDstyle=WS_VISIBLE; VC++下用MSComm控件实现串口通讯·2005-10-0817:59:47·来源:天极网m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); //串口控件的初始化 DWORDstyle=WS_VISIBLE; m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1); if(m_MSComm.GetPortOpen())//如果串口是打开的,则行关闭串口 { m_MSComm.SetPortOpen(FALSE); } m_MSComm.SetCommPort(1);//选择COM1 m_MSComm.SetInBufferSize(1024);//接收缓冲区 m_MSComm.SetOutBufferSize(1024);//发送缓冲区 m_MSComm.SetInputLen(0);//设置当前接收区数据长度为0,表示全部读取 m_MSComm.SetInputMode(1);//以二进制方式读写数据 m_MSComm.SetRThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件 m_MSComm.SetSettings("9600,n,8,1");//波特率9600无检验位,8个数据位,1个停止位 if(!m_MSComm.GetPortOpen())//如果串口没有打开则打开 m_MSComm.SetPortOpen(TRUE);//打开串口 else m_MSComm.SetOutBufferCount(0); //控件事件的响应声明 //*.h //{{AFX_MSG(CGolfView) afx_msgBOOLOnComm(); DECLARE_EVENTSINK_MAP() //}}AFX_MSG //*.cpp BEGIN_EVENTSINK_MAP(CGolfView,CView) //{{AFX_EVENTSINK_MAP(CAboutDlg) ON_EVENT(CGolfView,IDC_MSCOMM1,1/*OnComm*/,OnComm,VTS_NONE) //}}AFX_EVENTSINK_MAP END_EVENTSINK_MAP() //控件事件的响应 BOOLCGolfView::OnComm() { VARIANTvariant_inp; COleSafeArraysafearray_inp; LONGlen,k; BYTErxdata[2
m_Array.SetAt(i,m_SendData.GetAt(i));
m_mscomm.SetOutput(COleVariant(m_Array);
收数据:
给串口控件添加消息
OnOnCommMscomm()
VARIANTm_input;
char*str,*str1;
intk,nEvent,i;
CStringstr2,m_RcvData;
nEvent=m_MSComm.GetCommEvent();
switch(nEvent)
case2:
k=m_MSComm.GetInBufferCount(); //接收缓冲区的字符数目
if(k>0)
m_input=m_MSComm.GetInput();
str=(char*)(unsignedchar*)m_input.parray->pvData;
i=0;
str1=str;
while(i
m_disp+=m_RcvData;
UpdateData(false);
首先,在对话框中创建通信控件,若Control工具栏中缺少该控件,可通过菜单Project-->AddtoProject-->ComponentsandControl插入即可,再将该控件从工具箱中拉到对话框中。
此时,你只需要关心控件提供的对Windows通讯驱动程序的API函数的接口。
换句话说,只需要设置和监视MSComm控件的属性和事件。
打开所需串口后,需要考虑串口通信的时机。
在接收或发送数据过程中,可能需要监视并响应一些事件和错误,所以事件驱动是处理串行端口交互作用的一种非常有效的方法。
使用OnComm事件和CommEvent属性捕捉并检查通讯事件和错误的值。
发生通讯事件或错误时,将触发OnComm事件,CommEvent属性的值将被改变,应用程序检查CommEvent属性值并作出相应的反应
//若是在SDI中使用该控件则要调用下两句,在对话框程序中该语句有MFC自己创建
//所以不用人为添加
VC++下用MSComm控件实现串口通讯
·2005-10-0817:
59:
47·来源:
天极网
m_MSComm.Create(NULL,style,CRect(0,0,0,0),this,IDC_MSCOMM1);
//串口控件的初始化
//控件事件的响应声明
//*.h
//{{AFX_MSG(CGolfView)
afx_msgBOOLOnComm();
DECLARE_EVENTSINK_MAP()
//}}AFX_MSG
//*.cpp
BEGIN_EVENTSINK_MAP(CGolfView,CView)
//{{AFX_EVENTSINK_MAP(CAboutDlg)
ON_EVENT(CGolfView,IDC_MSCOMM1,1/*OnComm*/,OnComm,VTS_NONE)
//}}AFX_EVENTSINK_MAP
END_EVENTSINK_MAP()
//控件事件的响应
BOOLCGolfView:
OnComm()
VARIANTvariant_inp;
COleSafeArraysafearray_inp;
LONGlen,k;
BYTErxdata[2
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1