CSerialPort用法

在程序中如果要用到多个串口,而且还要做很多复杂的处理,那么最好不用MSComm通讯控件,如果这时你还不愿意自己编写底层,就用这个类:CserialPort类。
作者是 Remon Spekreijse ,可在http://www.codeguru.com/cpp/i-n/network/serialcommunications/article.php/c2483/A-communication-class-for-serial-port.htm找到作者的基于对话框的可以同时检测4个串口的通信例子.
本文介绍基于文档的程序中的用法:(实例为计算机上两个串口之间的发送与接收)
编程步骤:
◆1. 建立程序:
建立一个基于单文档的MFC应用程序SCPortTest,所有步骤保持缺省状态。

◆2. 添加类文件:
将SerialPort.h SerialPort.cpp两个类文件复制到工程文件夹中,并加入工程。并在视图类的头文件中包含:#include "SerialPort.h"。

◆3. 人工增加串口消息响应函数:OnCommunication(WPARAM ch, LPARAM port)
首先在视图类头文件中添加串口字符接收消息WM_COMM_RXCHAR(串口接收缓冲区内有一个字符)的响应函数声明:
//{{AFX_MSG(CSCPortTestView)
afx_msg LONG OnCommunication(WPARAM ch, LPARAM port);
//}}AFX_MSG

然后在视图类的cpp文件中进行WM_COMM_RXCHAR消息映射:
BEGIN_MESSAGE_MAP(CSCPortTestView, CView)
//{{AFX_MSG_MAP(CSCPortTestView)
ON_MESSAGE(WM_COMM_RXCHAR, OnCommunication)
//}}AFX_MSG_MAP
END_MESSAGE_MAP()

并且在本文件中加入函数的实现:
LONG CPortTestView::OnCommunication(WPARAM ch, LPARAM port)
{ ….. }

注意:由于这个串口类加入工程后,没有自动的消息映射机制,因此上述步骤均需要手工添加。

◆4 初始化串口
在视创建时初始化串口,首先利用ClassWizardr生成OnInitialUpdate()函数。
接着在SerialPort.h文件中说明我们在程序中要用到的全局变量:
保存两个串口接收数据:
char m_chChecksum; //用于COM1的校验和计算
CString m_strRXhhCOM1; //用于存放COM1接收的半BYTE校验字节hh
CString m_strRXDataCOM1; //COM1接收数据
CString m_strRXDataCOM2; //COM2接收数据
UINT m_nRXErrorCOM1; //COM1接收数据错误帧数
UINT m_nRXErrorCOM2; //COM2接收数据错误帧数
UINT m_nRXCounterCOM1; //COM1接收数据错误帧数
UINT m_nRXCounterCOM2; //COM2接收数据错误帧数CString

再在SerialPort.h文件中说明串口类对象:CSerailPort m_ComPort[2]; (public)。
因为要初始化2个串口,所以这里用了数组。
下面是初始化串口1和串口2:

void CSCPortTestView::OnInitialUpdate()
{
    CView::OnInitialUpdate();
    // TODO: Add your specialized code here and/or call the base class
    m_chChecksum=0; //校验和置0
    m_nRXErrorCOM1=0; //COM1接收数据错误帧数置0
    m_nRXErrorCOM2=0; //COM2接收数据错误帧数置0
    m_nRXCounterCOM1=0; //COM1接收数据错误帧数置0
    m_nRXCounterCOM2=0; //COM2接收数据错误帧数置0
    m_strRXhhCOM1.Empty(); //清空半BYTE校验hh存储变量
    for(int i=0;i<2;i++)
    {
        if (m_ComPort.InitPort(this,i+1,9600,‘N‘,8,1,EV_RXFLAG | EV_RXCHAR,512))
        //portnr=1(2),baud=960,parity=‘N‘,databits=8,stopsbits=1,
        //dwCommEvents=EV_RXCHAR|EV_RXFLAG,nBufferSize=512
        {
            m_ComPort
.StartMonitoring(); //启动串口监视线程
            if(i==1) SetTimer(1,1000,NULL); //设置定时器,1秒后发送数据
        }
        else
        {
            CString str;
            str.Format("COM%d 没有发现,或被其它设备占用",i+1);
            AfxMessageBox(str);
        }
    }
}

◆5 利用ClassWizard生成CPortTestView的时间消息WM_TIMER响应函数:

void CSCPortTestView::OnTimer(UINT nIDEvent)
{
    // TODO: Add your message handler code here and/or call default
    int randdata=rand()?00; //产生9000以内的随机数
    CString strSendData;
    strSendData.Format("d",randdata);
    SendString(strSendData, 2); //串口2发送数据;
    CView::OnTimer(nIDEvent);
}

上面用到的SendString()需按如下方式生成:
在ClassView中单击鼠标右键,在环境菜单中选择Add Member Function:

void CSCPortTestView::SendString(CString &str, int Port)
{
     char checksum=0,cr=CR,lf=LF;
     char c1,c2;
     for(int i=0;i<str.GetLength();i++)
     checksum = checksum^str;
     c2=checksum & 0x0f; c1=((checksum >> 4) & 0x0f);
     if (c1 < 10)
       c1+= ‘0‘;
else c1 += ‘A‘ - 10;
     if (c2 < 10)
       c2+= ‘0‘;
else c2 += ‘A‘ - 10;
     CString str1;
     str1=‘$‘+str+"*"+c1+c2+cr+lf;
     m_ComPort[Port-1].WriteToPort((LPCTSTR)str1);
}

请注意上面函数中是如何生成校验码的,要切记的是发送的校验码生成方式和对方接收的校验检测方式要一致。

◆6 在OnCommunication(WPARAM ch, LPARAM port)函数中进行数据处理
说明:WPARAM、 LPARAM 类型是多态数据类型(polymorphic data type),在WIN32中为32位,支持多种数据类型,根据需要自动适应,这样程序有很强的适应性。在此我们可以分别理解为char和 integer 类型数据。
每当串口接收缓冲区内有一个字符时,就会产生一个WM_COMM_RXCHAR消息,触发OnCommunication函数,这时我们就可以在函数中进行数据处理,所以这个消息就是整个程序的"发动机"。
下面是根据本文最初提出的问题写出的处理函数:

LONG CSCPortTestView::OnCommunication(WPARAM ch, LPARAM port)
{
    static int count1=0,count2=0,count3=0;
    static char c1,c2;
    static int flag;
    CString strCheck="";

if(port==2) //COM2接收到数据
    {
        CString strtemp=(char)ch;
        if(strtemp=="Y")
        {
            m_nRXCounterCOM2++;
            CString strtemp;
            strtemp.Format("COM2: NO.d", m_nRXCounterCOM2);
            CDC* pDC=GetDC(); //准备数据显示
            pDC->TextOut(10,50,strtemp);//显示接收到的数据
            ReleaseDC(pDC);
        }
    }

if(port==1) //COM1接收到数据
    {
        m_strRXDataCOM1 += (char)ch;
        switch(ch)
        {
            case ‘$‘:
                m_chChecksum=0; //开始计算CheckSum
                flag=0;
                break;
            case ‘*‘:
                flag=2;
                c2=m_chChecksum & 0x0f; c1=((m_chChecksum >> 4) & 0x0f);
                if (c1 < 10) c1+= ‘0‘; else c1 += ‘A‘ - 10;
                if (c2 < 10) c2+= ‘0‘; else c2 += ‘A‘ - 10;
                break;
            case CR:
                break;
            case LF:
                m_strRXDataCOM1.Empty();
                break;
            default:
            if(flag>0)
            {
                m_strRXhhCOM1 += ch; //得到收到的校验值hh
                if(flag==1)
                {
                    strCheck = strCheck+c1+c2; //计算得到的校验值hh
                    if(strCheck!=m_strRXhhCOM1) //如果校验有错
                    {
                        m_strRXDataCOM1.Empty();
                        m_nRXErrorCOM1++; //串口1错误帧数加1
                    }
                    else
                    {
                        m_nRXCounterCOM1++;
                        if(m_strRXDataCOM1.Left(1)=="$") //接收数据的第一个字符是$吗?
                        {
                            char tbuf[6];
                            char *temp=(char*)((LPCTSTR)m_strRXDataCOM1);
                            tbuf[0]=temp[1]; tbuf[1]=temp[2];
                            tbuf[2]=temp[3]; tbuf[3]=temp[4];
                            tbuf[4]=0; //0表示字符串的结束,必要
                            int data=atoi(tbuf);
                            CString strDisplay1,strDisplay2;
                            strDisplay1.Format("NO. d: The reseived data is d",
                      m_nRXCounterCOM1,data);
                            strDisplay2.Format("Error Counter=d.",m_nRXErrorCOM1);
                            CDC* pDC=GetDC(); //准备数据显示
                            //int nColor=pDC->SetTextColor(RGB(255,255,0));
                            pDC->TextOut(10,10,strDisplay1);//显示接收到的数据
                            pDC->TextOut(30,30,strDisplay2);//显示错误帧数
                            //pDC->SetTextColor(nColor);
                            ReleaseDC(pDC);
                            }
                    CString str1="Y";
                    m_ComPort[0].WriteToPort((LPCTSTR)str1);//发送应答信号Y
                    }
                    m_strRXhhCOM1.Empty();
                }
                flag--;
              }
              else
                m_chChecksum ^= ch;
                break;
            }
        }
        return 0;
}

时间: 2024-10-10 23:25:24

CSerialPort用法的相关文章

一个串口类CSerialPort及其简单使用

一个挺好用的串口类:CnComm1.3.SerialPort.rar 简单用法: 1.定义成员:       CSerialPort m_SerialPort;          2.初始化:    m_SerialPort.SetBufferSize(1024,1024);   m_SerialPort.SetWnd(m_hWnd);  m_SerialPort.SetNotifyNum(DEF_IN_BYTE_SIZE);  if (m_SerialPort.IsOpen())  {    

js中获取时间new date()的用法

js中获取时间new date()的用法 获取时间:   var myDate = new Date();//获取系统当前时间 获取特定格式的时间: 1 myDate.getYear(); //获取当前年份(2位) 2 myDate.getFullYear(); //获取完整的年份(4位,1970-????) 3 myDate.getMonth(); //获取当前月份(0-11,0代表1月) 4 myDate.getDate(); //获取当前日(1-31) 5 myDate.getDay();

20.5 Shell脚本中的逻辑判断;20.6 文件目录属性判断;20.7 if特殊用法;20.8 20.9 cace判断(上下)

扩展: select用法 http://www.apelearn.com/bbs/thread-7950-1-1.html 20.5 Shell脚本中的逻辑判断 格式1:if 条件 ; then 语句; fi 1. 创建if1.sh测试脚本: [[email protected] ~]# vi if1.sh a=5,如果a大于3,满足这个条件,显示ok 添加内容: #!/bin/bash a=5 if [ $a -gt 3 ] then echo ok fi 2. 执行if1.sh脚本: [[e

20.1 Shell脚本介绍;20.2 Shell脚本结构和执行;20.3 date命令用法;20.4 Shell脚本中的变量

20.1 Shell脚本介绍 1. shell是一种脚本语言 aming_linux blog.lishiming.net 2. 可以使用逻辑判断.循环等语法 3. 可以自定义函数 4. shell是系统命令的集合 5. shell脚本可以实现自动化运维,能大大增加我们的运维效率 20.2 Shell脚本结构和执行 1. 开头(首行)需要加: #!/bin/bash 2. 以#开头的行作为解释说明: 3. 脚本的名字以.sh结尾,用于区分这是一个shell脚本 4. 执行.sh脚本方法有两种:

shell 中seq的用法 echo -n用法

用法:seq [选项]... 尾数 或:seq [选项]... 首数 尾数 或:seq [选项]... 首数 增量 尾数 从1循环到100的两种方法(bash 其它的shell没试过)for x in `seq 1 100`;do echo $x;donefor x in {1..100};do echo $x;done echo -n 不换行输出 $echo -n "123" $echo "456" 最终输出 123456 echo -e 处理特殊字符 若字符串中

sudo的用法

su -l user -C 'COMMAND' 是用user这个用户执行命令 我们一般使用sudo 这个命令 sudo [-u] user COMMAND sudo [-k] COMMAND 清除此前用户的密码. sudo的配置文件/etc/sudoers 配置项为 users    hosts=(runas)    commands users:可以是一个用户的名称也可以是一个组,也可以是一个别名 username #UID user_alias 用户别名的用法 User_Alias NETA

几招学会 Python 3 中 PyMongo 的用法

本文和大家分享的是Python3下MongoDB的存储操作相关内容,在看本文之前请确保你已经安装好了MongoDB并启动了其服务,另外安装好了Python的PyMongo库.下面进入正题,一起来看看吧,希望对大家学习Python3有所帮助. 连接MongoDB 连接MongoDB我们需要使用PyMongo库里面的MongoClient,一般来说传入MongoDB的IP及端口即可,第一个参数为地址host,第二个参数为端口port,端口如果不传默认是27017. import pymongo cl

11 css中分组选择符的用法

<!doctype html> <html> <head> <meta charset="utf-8"> <title>无标题文档</title> <style type="text/css"> h1,span{color:red;} a:hover{color:#2EE926;} /*分组选择符的用法*/ </style> </head> <body&

gawk 文本处理入门用法详集

awk笔记 gawk - pattern scanning and processing language 报告生成器,可进行格式化输出,文本处理三剑客之一,是基于sed和grep功能的扩展 一般用法格式: awk [options] 'program' FILE...     program: /regular/{print} 语句之间用分号分隔    print,printf 选项: -F:指明输入时用到的字段    -v var=value:指明自定变量 awk运作方式: 逐行读入文本,并