《单片机串口通信及测控应用实战详解》——6.3 PC端程序设计

  1. 云栖社区>
  2. 博客>
  3. 正文

《单片机串口通信及测控应用实战详解》——6.3 PC端程序设计

异步社区 2017-05-02 17:23:00 浏览1563
展开阅读全文

本节书摘来自异步社区《单片机串口通信及测控应用实战详解》一书中的第6章,第6.3节,作者 李江全,聂晶,梁习卉子,刘新英,更多章节内容可以访问云栖社区“异步社区”公众号查看。

6.3 PC端程序设计

6.3.1 采用Visual Basic实现

1.程序界面设计

运行VB 6.0,创建标准的工程项目文件,设计程序窗体。

(1)添加1个MSComm控件用于实现PC与单片机串口通信。

(2)添加2个按钮控件CommandButton,用于输出指令和关闭程序。

(3)添加3个文本控件TextBox,用于输入单片机地址、继电器状态和返回数据。

(4)添加5个标签控件Label,用于显示文本框功能等。

2.属性设置

程序窗体、控件对象的主要属性设置如表6-2所示。


screenshot

设计的程序界面如图6-4所示。


screenshot

3.编写程序代码

以下是实现PC与多个单片机串口通信的Visual Basic程序。

'PC发送数据01 11,表示01号单片机板继电器1和2打开
'PC发送数据01 00,表示01号单片机板继电器1和2关闭
'PC发送数据02 01,表示02号单片机板继电器1关闭,继电器2打开
'PC发送数据02 10,表示02号单片机板继电器1打开,继电器2关闭
'初始化串口
Private Sub Form_Load()
  MSComm1.CommPort = 1
  MSComm1.Settings = "9600,n,8,1"
  MSComm1.InputMode = 1
  MSComm1.SThreshold = 1
  MSComm1.RThreshold = 1
  MSComm1.PortOpen = True
End Sub
'状态输出
Private Sub Cmdout_Click()
  Dim uout1 As Variant                                  '单片机地址号
  Dim uout2 As Variant                                     '单片机继电器状态值
  uout1 = "&H" & Val(addrText.Text)
  uout2 = "&H" & Val(dataText.Text)
  MSComm1.Output = Chr(uout1) & Chr(uout2)    '将十六进制数据值发送给单片机
End Sub
'返回数据
Private Sub MSComm1_OnComm()
  Dim Inbyte() As Byte
  Dim buffer As String
  Dim strdata(10) As String
  Dim datanum As String
  Select Case MSComm1.CommEvent
    Case comEvReceive
      Inbyte = MSComm1.Input                              '读取单片机返回的数据串
      For i = LBound(Inbyte) To UBound(Inbyte)
        buffer = buffer + Hex(Inbyte(i)) + Chr(32)     '获得单片机返回数据串
      Next i
    Case comEvSend
  End Select
    For jj = 1 To 6
      strdata(jj) = Mid(Trim(buffer), jj, 1)             '从数据串中逐个取位
    Next jj
    If Len(Trim(buffer)) = 3 Then
      datanum = "0" & strdata(1) & strdata(2) & "0" & strdata(3)
    End If
    If Len(Trim(buffer)) = 4 Then
      datanum = "0" & strdata(1) & strdata(2) & strdata(3) & strdata(4)
    End If
  receiveText.Text = datanum                '显示单片机返回的数据串
End Sub
'关闭程序
Private Sub Cmdexit_Click()
   MSComm1.PortOpen = False
   Unload Me
End Sub

4.运行程序

程序设计、调试完毕后,运行程序。

PC通过串行口将十六进制数发送给多个单片机,驱动地址吻合的单片机继电器动作,并在数码管显示接收的数。单片机接收到数据后,返回原数据给PC。

如PC发送十六进制数据“01 11”,驱动1号单片机板继电器1和2打开,单片机返回十六进制数据“01 11”。

程序运行界面如图6-5所示。


screenshot

6.3.2 采用Visual C++实现

1.程序界面设计

程序界面设计与VB多单片机通信基本相同。设计的程序界面如图6-6所示。


screenshot

2.编写程序代码

以下是实现PC与多个单片机串口通信的VC++参考程序。

#include "stdafx.h"
#include "mcu.h"
#include "mcuDlg.h"
//串口初始化设置
BOOL CMcuDlg::OnInitDialog()
{
  m_Comm.SetCommPort(1);   //选择COM1
  m_Comm.SetInputMode(1); //输入方式为二进制方式    
  m_Comm.SetRThreshold(1); //参数1表示每当串口接收缓冲区中有多于或等于1个字符时将引发一个接收数据的OnComm事件
  m_Comm.SetSThreshold(1);//参数1表示当传输缓冲区完全空时将引发一个接收数据的OnComm事件
  m_Comm.SetSettings("9600,n,8,1"); 
  m_Comm.SetPortOpen(TRUE);//打开串口
  return TRUE;  
}
//发送数据
void CMcuDlg::OnOK() 
{
  CByteArray data;
  CString a,b;
  int len;
  UpdateData(TRUE);  //读编辑框内容
  len = Str2Hex(m_address+m_data,data); 
  m_Comm.SetOutput(COleVariant(data));  
}
//接受返回数据
void CMcuDlg::OnOnCommMscomm1() 
{
  VARIANT data;
  COleSafeArray data2;
  CByteArray datatemp;
  LONG k = 0;
  BYTE Inbyte[2]; //设置BYTE数组
  CString strtemp;  
  if(m_Comm.GetCommEvent() =  = 2) //事件值为2表示接收缓冲区内有字符
  {
    data = m_Comm.GetInput();  //读缓冲区
      data2 = data;  //VARIANT型变量转换为ColeSafeArray型变量
    data2.GetElement(&k,&Inbyte[0]);//获取回传数据,前两个字节
    k++;
    data2.GetElement(&k,&Inbyte[1]);

  }
  strtemp.Format("%s  %s",HexChar(Inbyte[0]),HexChar(Inbyte[1])); 
  m_receive = strtemp;
  UpdateData(false);
}
//十六进制转为字符串
char* CMcuDlg::HexChar(BYTE idata)
{
     char *temp = new char[3];
  char a,b;
  temp[2] = '\0';
  a = idata>>4;
  b = idata&0x0f;
  if(a<9)
    temp[0] = (a+48);
  else 
    temp[0] = (a+55);
  if(b<9)
    temp[1] = (b+48);
  else 
    temp[1] = (b+55);
  return temp;  
}
//将字符转换十六进制
char CMcuDlg::HexChar(char c)
{
  if((c> = '0')&&(c< = '9'))
    return c-0x30;
  else if((c> = 'A')&&(c< = 'F'))
    return c-'A'+10;
  else if((c> = 'a')&&(c< = 'f'))
    return c-'a'+10;
  else 
    return -1;
}
//把字符串转换为十六进制
int CMcuDlg::Str2Hex(CString str, CByteArray &senddata)
{
  int hexdata,lowhexdata;
   int hexdatalen = 0;
   int len = str.GetLength();
   senddata.SetSize(len/2);
   for(int i = 0;i<len;)
   {
     char lstr,hstr = str[i];
     if(hstr =  = ' ')
     {
       i++;
       continue;
     }
     i++;
     if(i> = len)
       break;
     lstr = str[i];
     hexdata = HexChar(hstr);    //高位转换
     lowhexdata = HexChar(lstr); //低位转换
     if((hexdata =  = 16)||(lowhexdata =  = 16))
       break;
     else 
       hexdata = hexdata*16+lowhexdata;
     i++;
     senddata[hexdatalen] = (char)hexdata;
     hexdatalen++;
   }
   return hexdatalen;
}
//关闭串口退出程序
void CMcuDlg::OnCancel() 
{
  m_Comm.SetPortOpen(false);//关闭串口
  CDialog::OnCancel();
}

3.运行程序

程序设计、调试完毕,运行程序。

PC通过串行口将十六进制数发送给多个单片机,驱动地址吻合的单片机继电器动作,并在数码管显示接收的数。单片机接收到数据后,返回原数据给PC。

如PC发送十六进制数据“02 11”,驱动2号单片机板继电器1和2打开,单片机返回十六进制数据“0211”。

程序运行界面如图6-7所示。


screenshot

网友评论

登录后评论
0/500
评论
异步社区
+ 关注