基于MFC的串口通信(Mscomm)

1、串口通信的概述:

串口是一种重要的通信资源,例如鼠标口、USB接口都是串口。串行端口是CPU和串行设备间的编码转换器。当数据从CPU经过端口发送出去的时候,字节数据会被转为串行的位,在接收数据时,串行的位被转换为字节数据。

(1)、串口通信的特点:

数据通信多采用串口技术,主要因为串口可以在现有的电话网络上进行数据传输。串口通信是按照数据一位一位的依次传输,所以一根传输线就可以完成数据交换,降低了通信成本。

(2)、串口通信的传输方式

串口通信按照数据流可以分为三种传输方式:单工通信、半双工通信、全双工通信。

单工通信:使用一根导线,数据只能从A发送到B

半双工通信:是用一根导线,数据可以从A发送到B,也可以从B发送到A。但是不能同时进行

全双工通信:俩根导线。允许通信双方在俩个方向同时进行数据传输。

(3)、通信方式

同步通信:接收方不必对每个字节进行起始和停止的操作,传输效率高。传输设备复杂,双方时钟允许误差小。可用于点对点之间的数据传输。

异步通信:以字符为单位进行数据传输,并且每个字符都有起始位和停止位的标记。允许各个字符之间有间隙,俩个字符之间的间隔不固定。异步通信的传输效率低,传输设备简单,并且只适用于点对点的数据传输。

2、利用Mscomm进行串口通信:

(1)、字符格式收发:

1)、初始化框架的ICON和发送EDIT的文本

初始化框架上的内容,应该位于框架类的构造函数中

CComDlg::CComDlg(CWnd* pParent /*=NULL*/)

初始化ICON:m_hIcon = AfxGetApp()->LoadIcon(IDI_ICON_EARTH); // 图标改成地球
    

  初始化发送EIDT的文本:  m_strSend = "My first SerialPortTool!";

2)、初始化下拉列表

初始化下拉列表位于入口函数中

入口函数:

CComDlg::OnInitDialog

初始化串口号,波特率,校验位,数据位,停止位 

    m_cboPort.SetCurSel(0);
    m_cboBaudRate.SetCurSel(4);
    m_cboCheck.SetCurSel(0);
    m_cboDataBit.SetCurSel(3);
    m_cboStopBit.SetCurSel(0);

3)、获取我们界面中的设置到CMscomm类中的对象中

UI的串口设置中的打开按钮,添加打开按钮事件。

UpdateData(TRUE); //将控件中的内容同步到变量中,我们操作控件就相当于操作变量。

    CString strOpen;//strOpen来获得按钮上面的文字内容。
    GetDlgItemText(IDC_BUTTON_OPEN, strOpen);
    
    if(strOpen == "打开")
    {    //1、执行打开串口操作

        m_mscomm.put_InBufferSize(1024); //接收缓冲区  
        m_mscomm.put_OutBufferSize(1024);//发送缓冲区   
        m_mscomm.put_InputLen(0);//设置当前接收区数据长度为0,表示全部读取  
        m_mscomm.put_InputMode(CMscomm::comInputModeBinary);//以二进制方式读写数据   
        m_mscomm.put_RThreshold(1);//接收缓冲区有1个及1个以上字符时,将引发接收数据的OnComm事件
        
        //2、获取我们界面中的设置到CMscomm类中的对象中
        //校验位,犹豫我们拼接字符串,但是校验位比较特殊,很多设备没有校验位所以我们不能直接使用value类型变量
        CString strCheck = getCheck();

        CString strSettings = m_strBaudRate + strCheck + m_strDataBit + m_strStopBit;
        //MessageBox(strSettings);
        //"9600,n,8,1" 和 "9600n81" 均对
        //m_mscomm.put_Settings(_T("9600,n,8,1"));//波特率9600,无检验位,8个数据位,1个停止位  
        m_mscomm.put_Settings(strSettings);

        //端口号
        //put_CommPort参数直接代表串口号为1,m_cboPort.GetCurSel() + 1目的增加了复用性。
        m_mscomm.put_CommPort(m_cboPort.GetCurSel() + 1);//索引从0开始

getCheck()获取当前的校验位

//步骤2、获取校验位
CString CComDlg::getCheck(void)
{CString strCheck;switch(m_cboCheck.GetCurSel()){case 0: strCheck = "n"; break;//无校验case 1:  strCheck = "o"; break;//基校验case 2: strCheck = "e"; break;//偶校验case 3: strCheck = "m"; break;case 4:strCheck = "s"; break;default:break;}return strCheck;
}

4)、真正打开串口的操作

我们点击打开串口的时候,调用put_PortOpen打开串口,并且要做一个异常捕获。

打击打开串口的操作,我们的打开按钮得更换文本内容为关闭

同时有一个bmp的资源图片也会随着串口的打开和关闭改变图片

try
        {
            m_mscomm.put_PortOpen(TRUE);//put_PortOpen(TRUE),参数为TRUE则打开串口,为FALSE则关闭串口
        }
        catch (CException* e)
        {
            MessageBox("端口不存在!", "打开串口", MB_ICONERROR);
            return;
        }
        
        SetDlgItemText(IDC_BUTTON_OPEN, _T("关闭")); //串口打开之后,设置按钮为“关闭”

        CBitmap bitmap;  // CBitmap对象,用于加载位图    
        HBITMAP hBmp;    // 保存CBitmap加载的位图的句柄   
  
        bitmap.LoadBitmap(IDB_BITMAP_GREEN);  // 将位图IDB_BITMAP1加载到bitmap   
        hBmp = (HBITMAP)bitmap.GetSafeHandle();  // 获取bitmap加载位图的句柄   
        m_picIndicator.SetBitmap(hBmp);    // 设置图片控件m_picIndicator的位图图片为IDB_BITMAP_RED 
    }
    else // 4、此时串口已经处于打开状态 执行关闭串口
    {
        m_mscomm.put_PortOpen(FALSE);//put_PortOpen(TRUE),参数为TRUE则打开串口,为FALSE则关闭串口  
        SetDlgItemText(IDC_BUTTON_OPEN, _T("打开")); //串口打开之后,设置按钮为“关闭”

        CBitmap bitmap;  // CBitmap对象,用于加载位图    
        HBITMAP hBmp;    // 保存CBitmap加载的位图的句柄   
  
        bitmap.LoadBitmap(IDB_BITMAP_RED);  // 将位图IDB_BITMAP1加载到bitmap   
        hBmp = (HBITMAP)bitmap.GetSafeHandle();  // 获取bitmap加载位图的句柄   
        m_picIndicator.SetBitmap(hBmp);    // 设置图片控件m_picIndicator的位图图片为IDB_BITMAP_RED  
    }

5)、发送2进制或者16进制数据

首先当我们点击发送按钮的时候,进行一个异常捕获,如果串口没有打开之间弹出对话框串口未打开,然后我们进行一个勾选框的判断,判断发送的是2进制还是十六进制数据,最后进行数据处理和发送

UpdateData(TRUE); //1、读取编辑框内容 ,内容写到控件中,然后同步到变量中//2、什么时候发送,发送的条件就是文本内容是“发送”CString strSend;GetDlgItemText(IDC_BUTTON_SEND, strSend);if(strSend == "发送"){	//3、执行串口发送操作try{CString strOrdered = "";if(1 == m_chkHexSend.GetCheck())//当勾选框被选中的时候{	//以十六进制发送hexToSend.RemoveAll();//清空数组//十六进制 41 42 43 41空格为一组,42空格为一组....。strOrdered = GetOrderedStr();//MessageBox(strOrdered);for(int i = 0; i< strOrdered.GetLength(); i += 2){CString strTemp = strOrdered.Mid(i, 2);char *p = strTemp.GetBuffer(2);hexToSend.Add(strtol(p, NULL, 16));}m_mscomm.put_Output(COleVariant(hexToSend));//发送的二进制转换成16进制//4、这里如果串口没有打开直接写数据程序会崩溃,可能有异常直接try catch}else{//没有选中以字符格式发送strToSend = m_strSend;m_mscomm.put_Output(COleVariant(strToSend));//发送数据}

这里我们如果发送的是十六进制数据进行一个消息处理

CString CComDlg::GetOrderedStr()
{
    CString str;

    int i = 0;
    int length = m_strSend.GetLength();
    for(i = 0; i < length - 1; i++)//"31 32 33 回车 34"   "34 31 327"    "34 31 32 7"    "32 33 3 34"
    {
        if(m_strSend.Mid(i, 1) != " " && m_strSend.Mid(i, 1) != "\r")
        {
            if(m_strSend.Mid(i+1, 1) !=  " " && m_strSend.Mid(i+1, 1) !=  "\r")
            {
                CString strTemp;
                strTemp = m_strSend.Mid(i, 2);
                str = str + strTemp;
                i++;
            }
            else
            {
                str = str + "0" + m_strSend.Mid(i, 1);
            }
        }

        if(m_strSend.Mid(i, 1) == "\r" && m_strSend.Mid(i+1, 1) == "\n") //碰到回车,也就是\n\r,则直接跳过\n\r这两个字符! 
            i++;     
    }    

    if(i == m_strSend.GetLength()-1 && m_strSend.Mid(i, 1) != " ")
        str = str + "0" + m_strSend.Mid(i, 1);  //该行防止"34 31 32 7"中7丢失
    //至此, "31323334"      "34313207"      "34313207"     "32330334"

    //MessageBox(str);

    return str;
}

6)、接收十六进制或者二进制数据

void CComDlg::OnCommMscomm()
{// TODO: 在此处添加消息处理程序代码UpdateData(TRUE); static unsigned int cnt=0;  VARIANT variant_inp;    COleSafeArray safearray_inp;   long len,k;    byte rxdata[1024]; //设置 BYTE 数组   CString strtemp; //通过声明这样一个字节数组,您可以在后续的代码中使用rxdata来存储从串口读取的二进制数据switch(m_mscomm.get_CommEvent())  //1、如果我们读取到事件{case CMscomm::comEvReceive: //值为 2 表示接收缓冲区内有字符  //2、从串口缓冲区读取数据m_mscomm.put_InputMode(CMscomm::comInputModeBinary);//规定二进制方式读取数据cnt++;   variant_inp = m_mscomm.get_Input();   //  variant_inp = m_mscomm.get_Input()这行代码用于从串口读取数据,并将读取的数据存储在variant_inp变量中safearray_inp = variant_inp;   //variant_inp中的数据转换为SAFEARRAY类型,并存储在safearray_inp变量中。len = safearray_inp.GetOneDimSize(); //得到有效的数据长度   //状态栏显示接收到的字符个数CString strReceiveNum;m_ulReceiveNum += len;strReceiveNum.Format("%d", m_ulReceiveNum);strReceiveNum = "接收:" + strReceiveNum;m_StatusBar.SetText(strReceiveNum, 2, 0);//SBT_POPOUT, SBT_NOBORDERS//3、接收数据for(k = 0; k < len; k++)    {   safearray_inp.GetElement(&k, rxdata + k);if(1 == m_chkHexReceive.GetCheck()) //接收到的数据以十六进制显示{//safearray_inp.GetElement(&k, rxdata + k);  CString strtemp = "";strtemp.Format(_T("%02X"),rxdata[k]);//rxdata[k] 的值以十六进制格式添加到 strtemp 字符串中m_strReceive = m_strReceive + strtemp + " "; }else	//接收到的数据以字符格式显示{CString strtemp = "";//safearray_inp.GetElement(&k, rxdata + k);    strtemp.Format("%c",rxdata[k]); //将字符送入临时变量strtemp存放m_strReceive = m_strReceive + strtemp;//m_strReceive接收EDIT的变量}	}if (1 == m_chkHexReceive.GetCheck()) // 如果以十六进制显示,最后再加上一个换行{m_strReceive += "\r\n";}else{m_strReceive += "\n";}break;}UpdateData(FALSE); //更新编辑框内容//注意:在更新完编辑框的内容之后,还要设置接收编辑框定位到最后一行int nLineCount = m_editReceive.GetLineCount();int nLineLength = m_editReceive.LineLength(nLineCount);m_editReceive.LineScroll(nLineCount, nLineLength);//TRACE("%d", nLineCount);
}

7)、点击十六进制发送的时候,EDIT的内容变成十六进制

//将发送的文本内容直接从二进制编程变成16进制
void CComDlg::OnBnClickedCheckSendHex()
{	// TODO: 在此添加控件通知处理程序代码UpdateData(TRUE);if (1 == m_chkHexSend.GetCheck()) //此时要把字符格式转换成十六进制{char *p = m_strSend.GetBuffer(m_strSend.GetLength());m_strSend.ReleaseBuffer();CString str = "";int length = m_strSend.GetLength();for(int i = 0; i < length; i++){CString strTemp;strTemp.Format("%02X", p[i]);str = str + strTemp + " ";}m_strSend = str.TrimRight(" ");}else	//此时要把十六进制转换成字符格式{int length = m_strSend.GetLength();CString str;for(int i = 0; i< length; i += 3){CString strTemp = m_strSend.Mid(i, 2);char *p = strTemp.GetBuffer(2);int num = strtol(p, NULL, 16);strTemp.Format("%c", num);str = str + strTemp;}m_strSend = str;}UpdateData(FALSE);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/176043.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

用Visual Studio(VS)开发UNIX/Linux项目

目录 FTP是免不了的 正确设置头文件 组织项目结构 创建何种项目类型 FTP自动上传 大部分具有Windows开发经验的程序员会比较喜欢使用Visual Studio&#xff0c;而大部分Unix/Linux程序员则喜欢使用UltraEdit直接在主机上写代码。 为什么直接在主机上写代码呢&#xff0c;因…

AIGC - Qwen大模型:Qwen-7B模型推理部署

硬件环境 作为AIGC方面的小白来说&#xff0c;我抱着非常天真的想法&#xff0c;想让它在我的工作笔记本上用i5的CPU去跑&#xff0c;至于为什么这么想&#xff0c;当然是因为我没有GPU&#xff0c;身边也没有其他的带显卡电脑 恰好&#xff0c;在腾讯云看到了GN7的显示优惠活…

内存DMA及设备内存控制详解

序言 对于PCIe 设备&#xff08;PCIe Endpoint&#xff09;来说&#xff0c;其和CPU CORE、DRAM 的交互&#xff0c;主要涉及两种类型的内存访问&#xff1a; 设备内存访问&#xff1a;PCIe 设备的 Device Memory&#xff08;设备内存&#xff09;的访问&#xff0c;例如CPU …

③ 软件工程CMM、CMMI模型【软考中级-软件设计师 考点】

个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的路上摸爬滚打&#xff0c;记录学习的过程~ 个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ ③ 软件工程CMM、CMMI模型【软考中级-软件设计…

flink 反压原理

背景 在flink中由于数据倾斜或者数据处理速率的不匹配&#xff0c;很容易引起反压&#xff0c;本文就看一下flink反压的原理 flink反压原理 flink全流程pineline的反压实现其实依赖于TaskManager之间的反压和TaskManager内部的反压来实现 1.TaskManager之间的反压 2.Task…

视频下载软件 Downie4 mac中文介绍

Downie mac是一款Mac平台上非常实用的视频下载工具。它支持下载各种视频网站上的视频&#xff0c;并且具有快速、稳定、易于使用的特点。 Downie支持下载各种视频网站上的视频&#xff0c;包括YouTube、Vimeo、Netflix、Hulu、Amazon等等。它具有快速、稳定的下载速度&#xff…

Python---判定表法(功能测试)

能对多条件依赖关系进行设计测试点---判定表法 等价类、边界值分析法主要关注单个输入类条件的测试 定义:是一种以表格形式表达多条件逻辑判断的工具。 条件桩: 列出问题中的所有条件&#xff0c;列出条件的次序无关紧要动作桩: 列出问题中可能采取的操作&#xff0c;操作的…

python基于VGG19实现图像风格迁移

目录 1、原理 2、代码实现 1、原理 图像风格迁移是一种将一张图片的内容与另一张图片的风格进行合成的技术。 风格&#xff08;style&#xff09;是指图像中不同空间尺度的纹理、颜色和视觉图案&#xff0c;内容&#xff08;content&#xff09;是指图像的高级宏观结构。 实…

mac 安装homebrew ,golang

mac 安装homebrew ,golang 安装homebrew安装golang选择 apple arm 版本安装配置环境变量 安装homebrew /bin/zsh -c "$(curl -fsSL https://gitee.com/cunkai/HomebrewCN/raw/master/Homebrew.sh)"回车执行指令后&#xff0c;根据提示操作。具体包括以下提示操作&am…

每日一练 | 网络工程师软考真题Day46

阅读以下说明&#xff0c;答复以下【问题1】至【问题6】 【说明】 某公司总部效劳器1的操作系统为Windows Server 2003&#xff0c;需安装虚拟专用网〔VPN〕效劳&#xff0c;通过Internet与子公司实现平安通信&#xff0c;其网络拓扑结构和相关参数如图2-1所示。 【问题1】在Wi…

sql-50练习题16-20

sql-50练习题16-20 前言数据库表结构介绍学生表课程表成绩表教师表 1-6 检索"01"课程分数小于60&#xff0c;按分数降序排列的学生信息1-7 按平均成绩从高到低显示所有学生的所有课程的成绩以及平均成绩1-8 查询各科成绩最高分、最低分和平均分&#xff1a;以如下形式…

c++-set和map

文章目录 前言一、set容器1、set容器介绍2、set的使用2.1 set的构造函数和迭代器2.2 set的容量2.3 set修改操作 3、multiset容器3.1 multiset容器介绍3.2 multiset容器使用 二、map容器1、map容器介绍2、map容器使用2.1 map的构造函数与迭代器2.2 map中元素的修改2.3 map的容量…

Java修仙传之Flink篇

大道三千:最近我修Flink 目前个人理解&#xff1a; 处理有界&#xff0c;无界流的工具 FLINK&#xff1a; FLINK定义&#xff1a; Flink特点 Flink分层API 流的定义 有界数据流&#xff08;批处理&#xff09;&#xff1a; 有界流&#xff1a;数据结束了&#xff0c;程序也…

正则表达式包含数字和字符匹配

至少6位。 pattern : (?.[0-9])(?.[A-Za-z])[0-9A-Za-z]{6,} 正则表达式中的“?”是一个正向预查字符&#xff0c;它的意思是匹配前一个字符出现的最少一次。具体来说&#xff0c;当一个匹配出现时&#xff0c;它会检查前一个字符是否符合要求&#xff0c;如果符合&#xf…

【Java 进阶篇】深入理解 Java Response:从基础到高级

HTTP响应&#xff08;Response&#xff09;是Web开发中的一个关键概念&#xff0c;它是服务器向客户端&#xff08;通常是浏览器&#xff09;返回数据的方式。理解如何在Java中处理和构建HTTP响应是开发Web应用程序的重要一部分。本文将从基础知识到高级技巧&#xff0c;详细介…

ardupilot开发 --- 深度相机 篇

1. ZED 相机 1.1 规格 2. RealSense 需要机载计算机作为中介&#xff01;&#xff01;

分布式锁-Redis红锁解决方案

一 分布式锁的概念 1&#xff1a;概念 分布式锁&#xff08;多服务共享锁&#xff09; 在分布式的部署环境下&#xff0c;通过锁机制来让多客户端互斥的对共享资源进行访问控制分布式系统不同进程共同访问共享资源的一种锁的实现。如果不同的系统或同一个系统的不同主机之间共…

使用 Authing 快速实现一套类似 OpenAI 的认证、API Key 商业权益授权机制

如果你有经常使用 OpenAI 或者 HuggingFace 这一类面向开发者的 SaaS 服务&#xff0c;对于 API Key 肯定不会陌生。我们在使用这些服务时&#xff0c;通常都会在其平台上面创建一套 API Key&#xff0c;之后我们才能在代码中通过这一串 API key 访问其服务&#xff1b;同时&am…

处理SAP资产折旧AFAB 过账报错:“科目 8019010100 要求一个成本会计分配”

会计在进行资产折旧AFAB时 报错如下所示&#xff1a; 原因分析&#xff1a; 折旧时没有把资产设置得成本中心带到过账凭证的成本中心字段中去。而资产中已经维护了成本中心了。 所以要在资产过账的科目分配中设置一下路径如下&#xff1a; 或者TCODE&#xff1a;ACSET科目设置这…

Jmeter(二十一):jmeter导入和导出接口的处理(超详细)

JMeter测试导入接口 利用Jmeter测试上传文件&#xff0c;首先可根据接口文档或者fiddler抓包分析文件上传的接口&#xff1b;如下图&#xff1a; 以下是我通过fiddler所截取的文件上传的接口 1、填写导入接口的信息 查看文件上传栏下的填写信息&#xff1a; 文件名称&#x…