VC串口上位机简单例程源码及详细步骤_第1页
VC串口上位机简单例程源码及详细步骤_第2页
VC串口上位机简单例程源码及详细步骤_第3页
VC串口上位机简单例程源码及详细步骤_第4页
VC串口上位机简单例程源码及详细步骤_第5页
已阅读5页,还剩20页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第页VC++串口上位机简单例程(源码及详细步骤)VC++串口上位机简单例程.rar(4.33MB)

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口编号可以通过如下方式修改:

当然也可以通过上位机软件编写,通过按钮来选择COM端口号,但是此次仅仅是简单的例程,就没有弄那么复杂了。COM1口可用的话,会提示串口初始化完毕。否则会提示串口已经打开Portalreadyopen,表示串口已经打开,被占用了。

2、点击开始转换,串口会向单片机发送0xaa,单片机串口中断接收到0xaa后启动ADC转换一次,并把转换结果ADCL、ADCH共两个字节的结果发送至PC,PC进行数值转换后在窗口里显示。(见文章末尾图)

3、为防止串口被一只占用,点击关闭串口可以关闭COM1,供其它程序使用,点击后按钮变为打开串口,点击可重新打开COM1。

程序的编写:

1、打开VC++6.0建立基于对话框的MFC应用程序Test,

2、在项目中插入MSComm控件:工程->增加到工程->ComponentsandControls->双击RegisteredActiveXControls->选择MicrosoftCommunicationsControl,version6.0->Insert,按默认值添加,你会发现多了个电话图标,这是增加后串口通信控件。

3、删除确认、取消和提示框,添加“电话”、进程、静态文本、按钮、编辑框,拖动添加的控件,根据喜好布局。

4、右击编辑框Edit选择属性,在样式里设置,勾选多行、垂直滚动,其它可按默认值。

右击静态文本Text选择属性,在常规设置里,修改标题。

右击按钮PushButton选择属性,在在常规设置里,修改标题。

修改后界面如下,程序写出来运行时“电话”标志会自动消失。

5、查看->建立类向导MFCClassWizard->MemberViariable,选择ClassName为CTestDlg的类,ControlID为MSCOMM1,双击它,为它添加控制变量m_comm1。

类似的,选择IDC_BUTTON2添加控制变量m_serial。

(建立类向导也可以右击然后在弹出的快捷菜单里选择建立类向导)

至此,基本框架已经出来了,编译后运行可以看到如下所示的界面。(组建->全部组件,然后组建->执行)

6、点击左侧的视图窗口,可以在三种模式下切换,第三个是打开我们的源代码窗口,第一个是类,第二个是窗体的资源视图。

选择FileView,展开testfiles->HeaderFiles,打开testDlg.h,在全局变量下添加如下代码,然后保存:

intgllen;//定义整型标量gllen,用于记录接收数据的个数

CProgressCtrl*pbar;//指向进度条的指针,用于操作进度条

CStringstrRXDdata;//编辑框显示的文本,记录历次转换值

7、点击RecourseView,展开testrecourses->Dialog,双击IDD_TEST_DIALOG,编辑我们的主界面对话框。

双击击“电话”,弹出如下对话框,按确认键:

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<len;k++)

{

safearray.GetElement(&k,rxdata+k);//得到接接收到的数据放到BYTE型数组rxdata里

}

for(k=0;k<len;k++)

{

BYTEbt=(*(unsignedchar*)(rxdata+k));//读取AD转换的高字节

if((k%2)==0)

if((k+1)<len)

{

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);//更新文本控件的显示

这时重新编译一下,看会不会有什么错误,出现下面提示,可以选择全部组建来清除。

LINK:LNK4073:cannotcreatemapfor.ILKfile;linkingnonincrementally

出现下面错误,请关闭运行的test.exe后重试。

LINK:fatalerrorLNK1104:cannotopenfile"Debug/test.exe"

出现下面错误两种错误,是由于空间编号问题引起的,当我们添加了编辑框或者“电话”后再添加,其编号自动加一,就会出现控件没定义。

Z:\vc++串口上位机\test\testDlg.cpp(32):errorC2065:'IDC_MSCOMM1':undeclaredidentifier

Z:\vc++串口上位机\test\testDlg.cpp(139):errorC2065:'IDC_EDIT1':undeclaredidentifier

解决方法是,在RecourseView里,打开窗体IDD_TEST_DIALOG,右击“电话”或者编辑框等其它出错的控件,右击选择属性,在常规里修改ID,这里的程序,除BUTTON有1、2两个之外,其它都是1

全部组建编译一下,看看有没有错误,没有错误就可以运行一下,可以看到界面更原来是一样的。有错误就修改一下,省得弄多了,错在哪里都不知道,查起来麻烦。

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("关闭串口");//按钮显示状态改变

可以看到,串口的参数等等都在在这里初始化的,可以根据自己的需要修改的,具体可以查看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型,必须强制转换后才能发送

同样地,双击另外一个按钮,给串口操作按钮添加代码,用于关闭或者打开串口。添加如下代码:

if(!m_comm1.GetPortOpen())//判断串口是否已经打开

{

m_comm1.SetPortOpen(TRUE);//如果串口是关闭的,则打开串口

m_serial.SetWindowText("关闭串口");//按钮显示状态改变

}

else

{

m_comm1.SetPortOpen(FALSE);//如果已经打开串口,则关闭串口

m_serial.SetWindowText("打开串口");//按钮显示状态改变

}

至此,一个简单的串口上位机软件编写完成了,可以用来测试下,通过单片机往串口里发送数据,可以看到主窗口的的转换结果,已经进度条显示电压值变化。要把这个程序拿出来用,只需把…\vc++串口上位机\test\Release的test.exe拷出来用就行。Release可以在编译窗口里选择win32release,然后重新编译一下就出来了。

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个停止位

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: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<count;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控件实现串口通讯http://tech.163/school·2019-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/*OnCom

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论