zoukankan      html  css  js  c++  java
  • Linux下串口操作之数据拼接

    串口操作中,特别以非阻塞的方式读取和发送数据,做好进程之间的同步很重要。有时我们会发现这样一个问题,在进行read操作时,一次read不能获得一个完整的数据帧,这就好比你买了一个电脑,送货的先把显示器送到你家,再把机箱送到,你会发现还少键盘鼠标什么的,又要过几天才送,这会让你急死。很不幸,在串口操作的时候,接受数据很可能就是这样分批收货的,但是幸运的是,接受数据的动作很快,别忘了计算机就是靠速度这一点,抛开这个,啥都不是。

    很自然的,我们就会进行数据的拼接,将一堆零散的数据拼接成一个个有用的数据帧,哈哈,变废为宝。说多了让人很烦,举个例子吧。

    假如我们定义的数据帧是以'$'开头,以‘#’结尾的。

    首先定义了两个字符数组:

    //一个缓冲数组,用来存放每一次读到的数据
    char read_data[256]={0};
    //存放一个完整的数据帧,以便处理
    char read_buf[256]={0};

    再看看凭借数据的函数是怎么样实现的:

    //得到了一个完整的数据帧
    void get_complete_frame()
    {
        char read_tmp[256]={0};
        int return_flag=0;
        int i;
        //存放读取到的字节数
        while(1)
        {
          if(read(fd, read_tmp, sizeof(read_tmp))>0)
          {
             //数据帧的拼接
             printf("read_tmp: %s
    ",read_tmp);
             for( i=0;i<strlen(read_tmp);i++)
             {
                  if(read_tmp[i]=='$')
                  {
                     memset(read_data,0,sizeof(read_data));
                     char tmp[5]={0};
                     tmp[0]=read_tmp[i];
                     strcat(read_data,tmp);
                  }
                  else if(read_tmp[i]=='#')
                  {
                     char tmp[5]={0};
                     tmp[0]=read_tmp[i];
                     strcat(read_data,tmp);
                     return_flag=1;
                     memset(read_buf,0,sizeof(read_buf));
                     //遇到帧尾,将read_data帧拷贝到read_buf中,以便处理
                     memcpy(read_buf,read_data,sizeof(read_data));
                  }
                  else
                  {
                      char tmp[5]={0};
                     tmp[0]=read_tmp[i];
                     strcat(read_data,tmp);
                  }
             }
             //有了一个完整的数据帧就返回处理
             if(return_flag==1)
                return;
          }
          else//读不到数据就返回,以便检查对方是否断线
            return;
         usleep(100000);
        }
    }

    从上面的代码中,我们可以看到,每一次从串口读取数据,将读到的数据放在read_tmp中。对这个数组进行逐个地字符分析,遇到帧头标志就清空缓冲数组read_data中,保证了缓冲数组中的数组都是以‘$’开头的;如果遇到了帧尾,哈,我们现在遇到有了一个完整的帧啦,可以去处理帧咯,将数据帧拷贝到read_buf中,程序直接对read_buf进行处理,处理之前别忘了帧尾后面的字符是新的一桢的开头部分,要把它们也保存下来。在程序中我们看到读不到数据就返回,如果不返回,这个函数就会一直运行,那么这样做的效果不就等价于阻塞操作了么,非阻塞就失去了其意义。

    大概就这样吧。


  • 相关阅读:
    设计模式之一(策略模式)
    电脑开机进入不了XP界面
    IBM X系列笔记本通过U盘安装系统方法全攻略
    DELPHI 访问其它电脑文件(局域网)
    笔记本维修小插曲 屏幕不亮处理方式
    delphi 笔记
    电脑小子的新婚夜
    如何得到动态链接库的输出函数(delphi tdump.exe)
    sql server重复数据处理
    如何使用jQuery向asp.net Mvc传递复杂json数据Filter篇
  • 原文地址:https://www.cnblogs.com/james1207/p/3255901.html
Copyright © 2011-2022 走看看