关于单片机连接ESP8266模块截取其回显内容

发布者:西周以前的最新更新时间:2020-09-03 来源: 51hei关键字:单片机  ESP8266模块  回显内容 手机看文章 扫描二维码
随时随地手机看文章

1,首先,ESP使用串口,跟51单片机通信,控制端可以使用手机,但是,51单片机和手机不在一个档次,那么在51单片机上需要分析操作。


2,ESP8266在接收或者发送数据时,会向串口发送回显内容,也就是提示信息,提示接收到了什么,发送了什么,所以,在单片机上做字符串的分析截取很重要,不然手机和单片机的通信质量大大下降,前期的wifi小车写项目的时候,这个问题困扰了我很久,后来进度过慢,我就折衷的使用了分析一个字符的方法,对小车进行控制,但是这样的处理方式很差。


3,前阵子终于把wifi小车做完了,虽然只是简单的行进和lcd显示以及避障,但也不能继续做下去了,需要去学新东西了,在这之前我需要把ESP用的完善了才能安心学下一步的东西,也为以后的项目铺了路。


分析思路:
1,首先uart串口的初始化不再详述,学习过就可以初始化成功。


2,然后,在串口的中断服务函数中,用temp承接SBUF中的回显数据,再在中断服务函数中立即做判断处理,该存的存,该舍弃的跳过,而且在中断服务函数中不能做延时较长的动作,所以只能使用令人头疼的标志位。


3,那么如何辨别什么时候是一整句的回显呢?因为回显并不回显,最多也只有rn(回车换行),那么如何判断什么时候开始接收一句话的第一个字符呢?我现在使用的方法是,把n作为接收一句回显的开始,把r作为接收一句回显的结束。这个过程通过设置标志位实现,较为简单。
0,CONNECT
+IPD,0,1:y
对应十六进制:30 2C 43 4F 4E 4E 45 43 54 0D 0A 0D 0A 2B 49 50 44 2C 30 2C 31 3A 79
r n r n
以上是回显的模板,可以根据这些内容理解出使用rn判断起始终止位置比较有效。


4,其次,现在比较重要的就是获取手机发来的数据,也就是+IPD,0,3:abc(0:客户连接号,3:字符串长度)这样的回显句子,需要截取下来,并存在单片机中,供其他函数使用,那么解析这个字符串就是在动态的过程中做的,所以时效性需要很高,需要立即分析完成并且处理完毕,不能先存下来再判断,因为没有字符串结束符,结束标志完全靠着:前面的3,也就是字符串长度来判断,这里连r都没有用,只能动态的判断分析获取。


5,以下为串口中断服务函数51程序
//中断服务函数,用于软复位
void uart_isr() interrupt 4
{
//loc用来对ret_msg全局变量给偏移,用来组装一个字符串
static unsigned char i = 0;
unsigned char temp;
ES = 0;
temp = SBUF;//temp不能被改变,因为软复位需要用到
/*ESP8266截取字符串部分*/
if(temp == 'n')//开始符
{
Rev_status = BEGIN;//设置开始接收
}
else if(temp == 'r')//结束符
{
Rev_status = END;//设置结束接收
}
else//出来rn以外的字符
{
/************************************************/
//专门用来接收IPD和CIFSR,结束接收时要把Rev_Str_status置为无效,再次进入下面的循环,检测第一个字符
if(Rev_Str_status == BEGIN)
{
if((str_rev_flag == END) &&(temp != ':') && (str_len_flag == BEGIN))//开始接收字符串长度
{
str_len = str_len * 10 + (temp - '0');
}
//为了充分保证,只有一种情况,并且进入一次,采用多个flag
if((str_rev_flag == END) &&(First_dou_flag == OK) && (temp == ',') && (client_num_flag == BEGIN) && (str_len_flag == END))//再一次接收到了逗号,开始接收字符串长度
{
str_len_flag = BEGIN;
client_num_flag = END;
}
//第一个逗号来临,进入,以后不再进入
if((str_rev_flag == END) &&(First_dou_flag == NO) && (temp == ',') && (client_num_flag == END))//逗号来临
{
client_num_flag = BEGIN;
First_dou_flag = OK;//是第一个逗号
}
//开始接收用户连接号
if((str_rev_flag == END) && (temp != ',') && (client_num_flag == BEGIN))
{
client_num = client_num * 10 + (temp - '0');
}
//开始接收字符串
if(str_rev_flag == BEGIN)
{
//保存字符串到全局变量中,以便后面输出
Get_str[Get_str_loc] = temp;
Get_str_loc ++;
//如果字符串的长度和刚刚接收到的指明字符串长度相同,则不再接收,做收尾工作
if(Get_str_loc == str_len)
{
Get_str[Get_str_loc] = '';
Str_Ready = OK;//设置标志位,说明我已经接收到了一个整的字符串了,可以进行操作了。
Rev_Str_status = END;//清空接收字符串标志位,使得可以再次进入下面switch循环
client_num_flag = END;//清空接收client_num标志位
str_len_flag = END;//清空接收str_len字符串长度标志位
str_rev_flag = END;//清空接收真正字符串标志位
First_dou_flag = NO;//清空区分第几个逗号的标志位
}
}
if(temp == ':')//要是开始了:,那么后面开始接收字符长度为str_len的字符串长度
{
str_rev_flag = BEGIN;
}
}
/************************************************/
/************************************************/
//要在接收字符串的标志位无效,并且接收状态位有效的时候才做
if((Rev_Str_status == END) && (Rev_status == BEGIN))
{
//检测到第一个以后,立刻置接收标志位无效暂时不接收
switch(temp)
{
case '+'://要么接收到+IPD 要么接收到 +CIFSR
{
Rev_Str_status = BEGIN;//开始接收,暂时不进入这个switch循环
Rev_status = END;
break;
}
case 'E'://发送失败,回显ERROR
{
Send_flag = NO;
Rev_status = END;
break;
}
case 'S'://发送成功,回显SEND OK
{
Send_flag = OK;
Rev_status = END;
break;
}
default:
{
Rev_status = END;
break;
}
}
}
}
//软复位时使用
if(0x7f == temp)//special for Doflye
{
i ++;
if(10 == i)
{
i = 0;
ISP_CONTR = 0xe0;
}
}
else
{
i = 0;
}
RI = 0;
ES = 1;
}
6,由于是在动态过程中做的,那么判断和标志位是无可奈何的选择,这里就需要逻辑搞清楚,虽然上面的程序我也是比较晕的,但是凭着感觉写出来,在进行一点一点调试,最后还是能够稳定的实现的。算法方面我还需要加强。
7,还有一个就是要善用sprintf它可以把你需要显示的数字转化为对应的字符串。注意:最好使用int型。
8.下面是整个模块函数及使用范例
/********************使用示例**********************************/
// /*测试ESP8266,目的:获得回显信息*/
// #include
// #include "delay.h"
// #include "WifiESP8266.h"
// #include "lcd.h"
// #include "uart.h"
// #include "delay.h"
// #include
// #include
// extern bit Ok_flag;
// extern bit Str_Ready;
// extern bit Rev_Str_status;
// extern bit str_rev_flag;
// extern bit flag;
// extern unsigned char Get_str[40];
// extern int client_num;
// extern int str_len;
// void main()
// {
// unsigned char temp_buf[16] = "Tmp:26 Hum:56";
// int i = 0;
// uart_init();
//
// lcd_init();
// lcd_clean();
// delay_ms(500);
//
// WifiESP8266_Init("ZTLTest","0123456789","8000");
//
// while(1)
// {
// if(OK == SendStrToClint(temp_buf,0,strlen(temp_buf)))
// //发送给连接号为0的客户,hhh,三个字节长度
// {
// lcd_clean();
// lcd_write_str(1,1,"SEND OK");
// }
// else
// {
// lcd_clean();
// lcd_write_str(1,1,"ERROR");
// }
//
// if(check_revStr() == OK)//也可以改成while在某个函数体中做
// {
// /*************写对获取的字符串的操作**************/
// sprintf(temp_buf,"%d %d %s",client_num,str_len,Get_str);
// lcd_write_str(0,0,temp_buf);
// /*****************************************/
//
//
// /*****************测试接收成功后,回复信息,可不加******************/
// // if(OK == SendStrToClint(Get_str,0,strlen(Get_str)))
// // //发送给连接号为0的客户,hhh,三个字节长度
// // {
// // lcd_clean();
// // lcd_write_str(1,1,"SEND OK");
// // }
// // else
// // {
// // lcd_clean();
// // lcd_write_str(1,1,"ERROR");
// // }
// /******************************************************/
//
//
// /**************************必须要加!********************************/
// clean_flag();//调用wifi模块中的清空函数
// /*******************************************************/
// }
//
//
// }
// }
/************************************************************/
/*
wifi模块ESP8266模块测试函数
作者:张天乐
起始时间:2016/1/19
功能:完成基本的连接等功能
修改时间:2016/1/28
功能:能够实现区分哪个客户端发来的什么数据
*/
#include
#include "WifiESP8266.h"
#include "uart.h"
#include "lcd.h"
#include "delay.h"
#include
#include
//接收状态默认为结束
bit Rev_status = END;//接收状态位
bit Rev_Str_status = END;//接收字符串状态位
bit Send_flag = NO;//发送信息到手机状态位
bit OK_flag = NO;//成功状态位
bit Str_Ready = NO;//数据准备好状态位
bit First_dou_flag = NO;//是不是第一个逗号
//获取的字符串内容
unsigned char Get_str[40];
unsigned char Get_str_loc = 0;
//客户连接号,以及字符串长度
int client_num = 0;
int str_len = 0;
bit client_num_flag = END;
bit str_len_flag = END;
bit str_rev_flag = END;
//检测是否接收到了数据,供其他函数调用,如果接收到了数据,那么,就供其他函数提取全局变量Get_str
//在其他函数接收完以后,还要调用清空函数,清空标志位,以便下一次使用
bit check_revStr()
{
if(Str_Ready == OK)
{
return OK;
}
return NO;
}
void clean_flag()
{
//清空操作
memset(Get_str,0,sizeof(Get_str));
client_num = 0;
str_len = 0;
Get_str_loc = 0;//清空一部分全局变量,以便下一次操作,Get_str要在外面函数接收完再清空
Str_Ready = NO;//清空接收数据准备好标志位
}
//发送字符串给给客户端连接号为num的客户端,内容为参数str
//str:发送内容,str_len:发送的字节数,client_num:客户端连接号,注意:参数均为字符串形式!
bit SendStrToClint(unsigned char *str,int client_num,int str_len)//有待加入字符串解析,用于判断是否发送成功
{
//用于拼接AT指令的buffer
unsigned char AT_temp[30] = {0};
//拼接成: "AT+CIPSEND=client_num,str_lenrn" 设置发送,=后面第一个参数是客户连接号即client_num,第二个是需要发送的字节数
sprintf(AT_temp,"AT+CIPSEND=%d,%drn",client_num,str_len);
uart_sendstr(AT_temp);//将封装完的AT指令发送出去
delay_ms(50);
//发送需要发送的字节
uart_sendstr(str);
delay_s(1);
//经过一秒以后,查看是否有发送成功标志位,要是发送成功了,那么就置Send_flag为有效
if(Send_flag == NO)
{
return NO;
}
else
{
//发送完成以后,还要清空标志位为无效,以便下次使用
Send_flag = NO;
return OK;
}
}
//wifi模块的初始化函数 wifi名字和wifi密码和端口名称,IP名固定为192.168.4.1
void WifiESP8266_Init(unsigned char *name,unsigned char *password,unsigned char *port)
{
//用于拼接AT字符串命令
unsigned char AT_tempBuf[50] = {0};
//发送AT指令,设置wifi模式等
uart_sendstr("AT+RSTrn");//重启
delay_s(1);
uart_sendstr("AT+CWMODE=2rn");//设置为AP模式,wifi模块当做路由器
delay_s(1);
//设置wifi名称和密码
sprintf(AT_tempBuf,"AT+CWSAP="%s","%s",11,4rn",name,password);
uart_sendstr(AT_tempBuf);
memset(AT_tempBuf,0,sizeof(AT_tempBuf));//用完清空
delay_s(1);
uart_sendstr("AT+RSTrn");//重启
delay_s(1);
uart_sendstr("AT+CIPMUX=1rn");//设为多路
delay_s(1);
//开始拼接带有端口号的字符串
sprintf(AT_tempBuf,"AT+CIPSERVER=1,%srn",port);
uart_sendstr(AT_tempBuf);//打开服务,需要拼接
memset(AT_tempBuf,0,sizeof(AT_tempBuf));//用完清空
delay_s(1);
}
//中断服务函数,用于软复位
void uart_isr() interrupt 4
{
//loc用来对ret_msg全局变量给偏移,用来组装一个字符串
static unsigned char i = 0;
unsigned char temp;
ES = 0;
temp = SBUF;//temp不能被改变,因为软复位需要用到
/*ESP8266截取字符串部分*/
if(temp == 'n')//开始符
{
Rev_status = BEGIN;//设置开始接收
}
else if(temp == 'r')//结束符
{
Rev_status = END;//设置结束接收
}
else//出来rn以外的字符
{
/************************************************/
//专门用来接收IPD和CIFSR,结束接收时要把Rev_Str_status置为无效,再次进入下面的循环,检测第一个字符
if(Rev_Str_status == BEGIN)
{
if((str_rev_flag == END) &&(temp != ':') && (str_len_flag == BEGIN))//开始接收字符串长度
{
str_len = str_len * 10 + (temp - '0');
}
//为了充分保证,只有一种情况,并且进入一次,采用多个flag
if((str_rev_flag == END) &&(First_dou_flag == OK) && (temp == ',') && (client_num_flag == BEGIN) && (str_len_flag == END))//再一次接收到了逗号,开始接收字符串长度
{
str_len_flag = BEGIN;
client_num_flag = END;
}
//第一个逗号来临,进入,以后不再进入
if((str_rev_flag == END) &&(First_dou_flag == NO) && (temp == ',') && (client_num_flag == END))//逗号来临
{
client_num_flag = BEGIN;
First_dou_flag = OK;//是第一个逗号
}
//开始接收用户连接号
if((str_rev_flag == END) && (temp != ',') && (client_num_flag == BEGIN))
{
client_num = client_num * 10 + (temp - '0');
}
//开始接收字符串
if(str_rev_flag == BEGIN)
{
//保存字符串到全局变量中,以便后面输出
Get_str[Get_str_loc] = temp;
Get_str_loc ++;
//如果字符串的长度和刚刚接收到的指明字符串长度相同,则不再接收,做收尾工作
if(Get_str_loc == str_len)
{
Get_str[Get_str_loc] = '';
Str_Ready = OK;//设置标志位,说明我已经接收到了一个整的字符串了,可以进行操作了。
Rev_Str_status = END;//清空接收字符串标志位,使得可以再次进入下面switch循环
client_num_flag = END;//清空接收client_num标志位
str_len_flag = END;//清空接收str_len字符串长度标志位
str_rev_flag = END;//清空接收真正字符串标志位
First_dou_flag = NO;//清空区分第几个逗号的标志位
}
}
if(temp == ':')//要是开始了:,那么后面开始接收字符长度为str_len的字符串长度
{
str_rev_flag = BEGIN;
}
}
/************************************************/
/************************************************/
//要在接收字符串的标志位无效,并且接收状态位有效的时候才做
if((Rev_Str_status == END) && (Rev_status == BEGIN))
{
//检测到第一个以后,立刻置接收标志位无效暂时不接收
switch(temp)
{
case '+'://要么接收到+IPD 要么接收到 +CIFSR
{
Rev_Str_status = BEGIN;//开始接收,暂时不进入这个switch循环
Rev_status = END;
break;
}
case 'E'://发送失败,回显ERROR
{
Send_flag = NO;
Rev_status = END;
break;
}
case 'S'://发送成功,回显SEND OK
{
Send_flag = OK;
Rev_status = END;
break;
}
default:
{
Rev_status = END;
break;
}
}
}
}
//软复位时使用
if(0x7f == temp)//special for Doflye
{
i ++;
if(10 == i)
{
i = 0;
ISP_CONTR = 0xe0;
}
}
else
{
i = 0;
}
RI = 0;
ES = 1;
}
最后,还是有点缺陷的是,没有很好的完成安卓的客户端,只能发送字符串给单片机,但是不能将单片机发来的数据用socket在手机上显示出来,安卓是个弱项,但是socket通信还是需要好好学学,网络编程后续学习,对应安卓客户端我也会尝试写出来。

[1] [1]
关键字:单片机  ESP8266模块  回显内容 引用地址:关于单片机连接ESP8266模块截取其回显内容

上一篇:用12C5A60S2和RF905写一个无线发射和接收程序用12自带的SPI
下一篇:STC8951单片机的外部中断资源

推荐阅读最新更新时间:2024-11-07 21:59

如何设计一个以AVR单片机为核心的安全工器具监控系统
众所周知,安全是电力企业一个永恒的主题,而的好坏,将直接影响着电力企业在生产过程中的人身和设备安全。笔者提出了一种基于的安全工器具的设计方案,主要是对安全工器具的离就位状态信息、有效使用期限和安全工器具柜内的温湿度状况等进行监控,可以根据实现安全工器具柜内温湿度的自动调节,能够自动记录安全工器具的相关信息。本文首先阐述了系统的总体结构,然后详细论述了系统的硬件设计和软件设计。 1 安全工器具监控系统的总体结构 1.1 系统结构 本文所设计的安全工器具监控系统主要分为下位机数据采集和上位机数据管理两个部分,两者之间通过网络进行通信。 系统首先通过数据采集板对安全工器具柜的相关量进行采集,然后将采集得到的数据进行相应处理后经串口服
[单片机]
如何设计一个以AVR<font color='red'>单片机</font>为核心的安全工器具监控系统
51单片机的状态迁移与复位操作
   1 单片机的工作状态及其状态迁移   80C51单片机的各种活动,可以描述成多个不同的工作状态或工作模式。这里利用笔者构思的一个单片机工作状态迁移图(如图1所示)来说明。不过,这里重点关注的是复位状态。 500)this.width=500" border=0> 图1 单片机工作状态迁移图   经过仔细分析之后,从图1中可以看出:   ① 把单片机经历的所有生存状态归纳和描绘成5个状态——1个非工作状态(即无电状态)和4个工作状态。   ② 只有复位状态是一个暂态,其他均为稳态;并且每次单片机进入正常运行状态时,都要经历一次复位状态。   ③ 只有在正常运行状态(这里记作NORMAL)下,单片机才按照程序存储器中固
[单片机]
51<font color='red'>单片机</font>的状态迁移与复位操作
利用小型MCU实现LED照明的色彩控制
中心议题 *分析了利用小型MCU实现LED照明的色彩控制的方法 解决方案 *MCU读取来自光传感器的颜色成分,校准光传感器输出,并通过调节各个LED驱动器的输出获得想要的颜色 *LED使用恒流驱动器来维持光输出的一致性 LED用于通用照明已指日可待。LED在通用照明系统中优势很多,如寿命更长以及效率更高。然而, LED技术还面临着一些挑战。其中一个挑战就是如何产生高品质的白光。白光LED的构成包含了蓝光LED 和能将光输出移至光谱的其他波段的一种荧光粉。许多白光LED都无法产生高显色指数(Color Rendering Index,CRI),该参数用于衡量光源真实重现色彩的能力。 通过混合两种或两种以
[电源管理]
利用小型<font color='red'>MCU</font>实现LED照明的色彩控制
PIC单片机之中断程序
什么是中断程序呢? 形象的生活比喻就比如你现在这在看我的文章,突然你的朋友喊你一起去烤地瓜,这时候你就中断了看文章和朋友烤地瓜去了,烤完地瓜之后你又回来看文章。烤地瓜这件事就好比中断程序,他中断了你看文章这件事。在程序方面来说 当CPU在执行一个程序的时候,突然产生了中断事件CPU就去执行中断程序了,当执行完成后CPU又回来执行原先的程序。 中断事件 什么是中断事件,就是引起中断的事件。对于单片机来说这些事件是多种多样的。比如 说一个按键按下,一定的时间到了,一串数据发送完毕,或接收完一个数据。 讲到中断不得不讲讲和中断相对的 查询。其实不管是按键按下 还是 时间到,还是数据发送完毕,这些事实上都可以用查询的方式办到。比如
[单片机]
基于MCU+FPGA模式的RFID读写器设计
射频识别技术RFID是一种非接触的自动识别技术,其基本原理是利用射频信号和空间耦合(电感和电磁耦合)传输特性,实现对被识别物体的自动识别,射频识别系统一般由两部分组成,射频标签(Tag)和射频读写器(Reader)。在RFID应用中,电子标签附着在被识别物体上,当带有射频标签的被识别物品进入读写器的可识读范围内,读写器自动以无接触方式将射频标签中约定的信息读取出来,从而实现自动识别物品和收集物品标志信息的功能。 RFID技术在生产、零售、物流和交通等行业有着广阔的应用前景。为解决后勤物资在请领、运输、分发等环节中的可视化动态监控问题,在军事物流领域广泛使用RFID技术,其基本用法是在仓库、码头、车站、重要交通路口安装布设射频读写
[嵌入式]
基于<font color='red'>MCU</font>+FPGA模式的RFID读写器设计
68HC908JB8单片机及其在USB外设系统中的应用
主要参数及引脚功能     MC68HC908JB8是采用HCMOS工艺技术生产的高性能单片机芯片,具有片内256BRAM和8KB的Flash ROM结构,16位双通道TIM模块(每一通道配有输入捕捉、输出比较和PWM工作模式),以及兼容USB1.1协议低速通信功能。 JB8 的性能     68HC908JB8是一种高性能M68HC08结构,它和M6805、M146805及M68HC05族代码兼容,3MHz内部总线频率,256B的RAM和8129字节的Flash存储器,且在片内可编程。最多可达37个一般3.3V的I/O口。其中包括:     ·因封装不同有13或10个共享I/
[应用]
基于AL101单片机控制的网络交换机的设计
随着计算机性能的提高及通信量的聚增,传统局域网已经越来越超出了自身负荷,交换式以太网技术应运而生,大大提高了局域网的性能。网络交换机能显著的增加带宽,可以建立地理位置相对分散的网络。局域网交换机的每个端口可并行、安全、实时传输信息,而且性能稳定、结构灵活、易于安装、便于管理,能很好地满足企业网和电信运营商宽带接入的需求。 1 网络交换机的硬件设计 随着人们对网络应用中的安全性和高带宽的需求,网络交换机的用途越来越广。本交换机采用了AL101芯片的ROX总线,将3个8口交换芯片连接起来,组成了1个24端口交换机,满足了用户对多交换端口的需求。 1.1 电路性能要求 交换机的高速PCB电路板,在EMC和ESD上都有比较高的要求
[工业控制]
基于AL101<font color='red'>单片机</font>控制的网络交换机的设计
8051单片机典型接口电路——串行扩展静态显示电路
8051单片机典型接口电路——串行扩展静态显示电路如下图所示: 例1:按上图编制显示子程序,显示字段码已分别存在32H~30H内RAM中。   程序1:   DIR2: MOV SCON,#00H ;置串口方式0   CLR ES ;串口禁中   SETB P1.0 ;“与”门开,允许TXD发移位脉冲   MOV SBUF,30H ;串行输出个位显示字段码   JNB TI,$ ;等待串行发送完毕   CLR TI ;清串行中断标志   MOV SBUF,31H ;串行输出十位显示字段码   JNB TI,$ ;等待串行发送完毕   CLR TI ;清串行中断标志   MOV SBUF,32H ;串
[模拟电子]
8051<font color='red'>单片机</font>典型接口电路——串行扩展静态显示电路
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved