我想实现的是,我用电脑在串口监视器上输入一个字符串,arduino能识别这个字符串中的每一个字符并在相应的串口上给出相应的高低电平以驱动舵机,比如输入L1,RS,功能是左手腕舵机逆时针旋转90°,然后右手指舵机松开魔方,鉴于arduino的Serial.read()函数一次只能从串口上读取一个字节,所以要想输入字符串,得先写读入字符串程序。网上这种程序有,复制粘贴一段:String comdata = "";
void setup()
{
    Serial.begin(9600);
}
void loop()
{
    while (Serial.available() > 0) 
    {
        comdata += char(Serial.read());
        delay(2);
    }
    if (comdata.length() > 0)
    {
        Serial.println(comdata);
        comdata = "";
    }
}此代码作用就是用Serial.read()以此读入数据,然后存在comdata字符串内,之后要调用只需要comdata[x](x为你指定的整型数字)即可,但是我加进我的程序之后,发现问题并不这么简单,怎么调都出问题,在无能为力的情况下,只能重新写程序,一点一点研究走。我自己重写了一个程序,用于研究。

首先,我认为问题出在串口上面,arduino的串口一次通过一个值,如果不通过电脑给arduino输出的话,它应该会通过一个默认值。写入程序,发现这个默认值为-1

那么我们就需要一个判断语句,把这些没用的值过滤掉。if(myinput > 0)即可。这里注意,Serial.read()语句调用一次读一个字符,调用第二次就读下一个字符,所以我们需要一个临时的变量存储Serial.read()读到的值,问题是,该用什么类型定义这个变量呢?int myinput;然后 myinput = Serial.read(); 为啥?这里重要的事情说三遍:Serial.read()从电脑端读入的值强制转换成整型输出!Serial.read()从电脑端读入的值强制转换成整型输出!Serial.read()从电脑端读入的值强制转换成整型输出!但是,你从电脑端输入的是字符型;你从电脑端输入的是字符型;你从电脑端输入的是字符型;也就是说,你要输入1,myinput就是1字符的ASC2码,就是49,输入2,myinput就是整型50,输入S,返回的就是83!所以,要把myinput减一个‘0’,让他变成整形数字,这样的话,1为1,2为2,S就是35。明白了这个,之后的操作才可行。

输入123JS,成功!不过串口默认值变成-4。

当然你也可以用char来处理,不过默认值就变成了不知道什么的字符,看着很难受。

解决的这个问题后,来解决下一个问题,如果我在电脑上输入两个字符LS,存进两个myinput里,但是机器怎么识别是不是有默认值混进来呢,因为电脑不传命令,就传默认值,所以到头来,我的myinput可能是-1L,或是S-1,这就不好了。没有关系,加入这段代码即可//判断输入指令  if(myinput1 > 0)  {    order1 = myinput1 - '0';    order2 = myinput2 - '0';  }  else  {    order1 = myinput2 - '0';    order2 = myinput3 - '0';   }意思是,从串口里读入三个信号,然后从中依次判别各个信号是否有效(是否大于0),最后输出有效字符,存入order内,从而剔除了默认值-1。串口的问题已解决,之后的问题都迎刃而解,程序一下次就实验成功了。下面给出arduino源码//arduino环境#include <string.h>int servopinL1=9;//定义数字接口9 连接左手旋转伺服舵机信号线int servopinL2=10;//定义数字接口10 连接左手夹持伺服舵机信号线int servopinR1=8;//定义数字接口9 连接右手旋转伺服舵机信号线int servopinR2=7;//定义数字接口10 连接右手夹持伺服舵机信号线int myinput1;int myinput2;int myinput3;int order1;int order2;int myangle;//定义角度变量int pulsewidth;//定义脉宽变量int val;String comdata = "";void servopulse(int servopin,int myangle)//定义一个脉冲函数{pulsewidth=(myangle*11)+500;//将角度转化为500-2480 的脉宽值digitalWrite(servopin,HIGH);//将舵机接口电平至高delayMicroseconds(pulsewidth);//延时脉宽值的微秒数digitalWrite(servopin,LOW);//将舵机接口电平至低delay(20-pulsewidth/1000);}void setup(){  Serial.begin(9600);//连接到串行端口,波特率为9600  Serial.println("servo1 ready" ) ;  Serial.println("servo2 ready" ) ;    pinMode(servopinL1,OUTPUT);//设定舵机接口为输出接口  pinMode(servopinL2,OUTPUT);//设定舵机接口为输出接口  pinMode(servopinR1,OUTPUT);//设定舵机接口为输出接口  pinMode(servopinR2,OUTPUT);//设定舵机接口为输出接口}void loop(){  myinput1 = Serial.read();  myinput2 = Serial.read();  myinput3 = Serial.read();  Serial.println(myinput1);  Serial.println(myinput2);  Serial.println(myinput3);  delay(1000);    //判断输入指令  if(myinput1 > 0)    if(myinput2 > '0')    {       order1 = myinput1 - '0';       order2 = myinput2 - '0';           }  else  {    order1 = myinput2 - '0';    order2 = myinput3 - '0';  }    if(order1 > 0 && order2 > 0)  {  Serial.println(order1);  Serial.println(order2); if(order1 == 28) {  switch (order2)  {    case 1:          val = 0;      val=val*(180/2);//将数字转化为角度      Serial.print("anticlockwise ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinL1,val);//引用脉冲函数      }     break;             case 2:          val = 1;      val=val*(180/2);//将数字转化为角度      Serial.print("no move ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinL1,val);//引用脉冲函数      }    break;            case 3:          val = 2;      val=val*(180/2);//将数字转化为角度      Serial.print("clockwise ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinL1,val);//引用脉冲函数      }     break;         case 26:       //字符串'J'的ASC二码转化为整形后 - '0'          val = 4;      val=val*(180/9);//将数字转化为角度      Serial.print("close ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinL2,val);//引用脉冲函数      }     break;         case 35:       //字符串'S'的ASC二码转化为整形后 - '0'          val = 2;      val=val*(180/9);//将数字转化为角度      Serial.print("open ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinL2,val);//引用脉冲函数      }     break;        }  Serial.println("Good job!");  myinput1 = 0;  myinput2 = 0;  myinput3 = 0;  order1 = 0;  order2 = 0;  }  else if(order1 == 34)  {      switch (order2)  {    case 1:          val = 0;      val=val*(180/2);//将数字转化为角度      Serial.print("anticlockwise ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinR1,val);//引用脉冲函数      }     break;             case 2:          val = 1;      val=val*(180/2);//将数字转化为角度      Serial.print("no move ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinR1,val);//引用脉冲函数      }    break;            case 3:          val = 2;      val=val*(180/2);//将数字转化为角度      Serial.print("clockwise ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinR1,val);//引用脉冲函数      }     break;         case 26:       //字符串'J'的ASC二码转化为整形后 - '0'          val = 5;      val=val*(180/9);//将数字转化为角度      Serial.print("close ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinR2,val);//引用脉冲函数      }     break;         case 35:       //字符串'S'的ASC二码转化为整形后 - '0'          val = 0;      val=val*(180/9);//将数字转化为角度      Serial.print("open ");      Serial.print(val,DEC);      Serial.println();      for(int i=0;i<=50;i++) //给予舵机足够的时间让它转到指定角度      {        servopulse(servopinR2,val);//引用脉冲函数      }     break;        }  Serial.println("Good job!");  myinput1 = 0;  myinput2 = 0;  myinput3 = 0;  order1 = 0;  order2 = 0;  }  }  }
---------------------

来源:CSDN
原文:https://blog.csdn.net/qq_26371477/article/details/79397613

arduino通信问题的学习与解决的更多相关文章

  1. 基于51单片机IIC通信的PCF8591学习笔记

    引言 PCF8591 是单电源,低功耗8 位CMOS 数据采集器件,具有4 个模拟输入.一个输出和一个串行I2C 总线接口.3 个地址引脚A0.A1 和A2 用于编程硬件地址,允许将最多8 个器件连接 ...

  2. MindSpore联邦学习框架解决行业级难题

    内容来源:华为开发者大会2021 HMS Core 6 AI技术论坛,主题演讲<MindSpore联邦学习框架解决隐私合规下的数据孤岛问题>. 演讲嘉宾:华为MindSpore联邦学习工程 ...

  3. 【IScroll深入学习】解决IScroll疑难杂症

    前言 在去年,我们对IScroll的源码进行了学习,并且分离出了一段代码自己使用,在使用学习过程中发现几个致命问题: ① 光标移位 ② 文本框找不到(先让文本框获取焦点,再滑动一下,输入文字便可重现) ...

  4. 清晰易懂TCP通信原理解析(附demo、简易TCP通信库源码、解决沾包问题等)C#版

    目录 说明 TCP与UDP通信的特点 TCP中的沾包现象 自定义应用层协议 TCPLibrary通信库介绍 Demo演示 未完成功能 源码下载 说明 我前面博客中有多篇文章讲到了.NET中的网络编程, ...

  5. win7下Arduino Mega 2560驱动安装失败解决办法

    因为玩四轴用的apm的飞控板,而其需要安装此驱动,曾经在win8使用其,但是因为win8有相对应的数字证书保护措施(应该是这样的,因为好久了记不清楚了),以至于我每次都需要长按shift重启电脑关闭此 ...

  6. ORA-03113:通信通道的文件结尾解决

    今天跟往常一样,登陆PL/SQL,确登陆失败,出现一个错误“ORA-01034”和“ORA-27101”如图: 然后就就通过命令提示符去登陆Oracle,去查看怎么回事,然后问题进一步出现,错误“OR ...

  7. PC-PC-单片机(Arduino)通信实例

    请仔细理解相关参数,如端口设置.IP设置.COM口设置......等等.....不要盲目COPY.....这涉及2台电脑和一个单片机,其中一台电脑作为服务器并与单片机相连,负责通过网络与客户端通信(s ...

  8. 自学了三天的SeaJs学习,解决了前端的一些问题,与小伙伴们一起分享一下!

    我为什么学习SeaJs? [第一]:为了解决项目中资源文件版本号的问题,以及打包压缩合并等问题. [第二]:好奇心和求知欲.[我发现很多知名网站也都在使用(qq空间, msn, 淘宝等等),而且 Se ...

  9. 基于51单片机IIC通信的AT24C02学习笔记

    引言 最近在学习几种串行通信协议,感觉收获很多,这篇文章是学习IIC总线协议的第一篇文章,以后还会再写一篇关于PCF8591 IIC通信的ADDA转换芯片的文章. 关于IIC总线 IIC 即Inter ...

随机推荐

  1. js实现超范围的数相加

    在js中能表示的最大安全整数是 9007199254740991,可以用API Number.MAX_SAFE_INTEGER 看一下  超出范围就会发生精度丢失,像这样 解决方法: 相当于一个字符串 ...

  2. luogu P3356 火星探险问题

    本题很简单的费用流问题,有石头的点需要限制,那我们就可以拆点,capacity为1就可以限制,然后cost为-1,直接跑板子就可以了,注意输出的时候找残量网络的反向边

  3. JavaScript一个简单的图片切换布局

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  4. FLASK - 请求和响应

    请求都封装在request对象里面了 from flask import Flask from flask import request from flask import render_templa ...

  5. LibreOJ #6001. 「网络流 24 题」太空飞行计划

    \(\quad\) 与网络流有关的最值有三个:最大流,最小费用,最小割.这道题是最小割.想了好久,终于想明白最小割应该怎么用. \(\quad\) 先找出矛盾的事物.在这道题中,两件事是矛盾的:做实验 ...

  6. JMeter配置JDBC测试SQL Server/MySQL/ORACLE

    一.配置SQL Server 1.下载sql驱动,将sqljdbc4.jar放到JMeter安装目录/lib下. 2.启动JMeter,右键添加->配置文件->JDBC Connectio ...

  7. 十一 队列 Queue

    队列:  一种先进先出的数据结构  FIFO 数组队列的实现:

  8. 怎样实现android 返回到上一个Activity并重新执行一次onCreate方法

    1.onCreate 方法只在activity一开始创建的时候执行.2.也就是在该activity销毁后才能再次执行,假如当前activity上再打开一个activity,并且原来的activity已 ...

  9. Ajax接收Json数据,调用template模板循环渲染页面的方法

    一. 后台接口吐出JSON数据 后台php接口中,需要写三个部分: 1.1 开头header规定数据格式: header("content-type:application/json;cha ...

  10. Q12问题

    1.我知道公司对我的工作要求吗? 2.我有做好我的工作所需要的材料和设备吗? 3.在工作中,我每天都有机会做我最擅长做的事吗? 4.在过去的七天里,我因工作出色而受到表扬吗? 5.我觉得我的主管或同事 ...