文章目录
- 前言
- 一、ESP8266调试
- 二、代码部分
- 1.esp8266部分
- 2.单片机代码部分
- 总结
前言
因为之前学习stm32+esp8266请求心知天气时看的不是很懂,然后在成功调试出来之后发一篇面向刚入门的学者,当然我也属于是刚入门级别的。
一、ESP8266调试
我这里使用的是ESP8266-01S,首先得烧录固件库,这些买的时候商家也会提供,所以不多做解释。
首先配置好ESP8266所用的指令
AT+CWMODE=1
AT+RST
AT+CWJAP=“232”,"123456789\” 这里的232是你的热点名称 后面数字代表密码
AT+CIPMUX=0 单连接模式
AT+CIPSTART=“TCP”,“116.62.81.138”,80 TCP为连接协议,后面为服务器IP,80表示端口
AT+CIPMODE=1 设置为透传模式
AT+CIPSEND 这个是用来发送数据前的AT指令,当使用完这个指令返回">"这个符号之后 esp8266就会禁止以后的AT指令发送
最后就是直接GET请求 以心知天气API为例:GET https://api.seniverse/v3/weather/now.json?key=你的私钥&location=zhaoqing&language=zh-Hans&unit=c
还有一点就是当发送了GET请求之后平台返回的数据是直接进入与esp8266相连的单片机的中断,不需要另外写接收代码
最后可以使用调试助手查阅是否获取到了数据,我这里用的是安信可:https://docs.ai-thinker/_media/tools/aithinker_serial_tool_v1.2.3.7z 直接点击链接既可下载
二、代码部分
1.esp8266部分
我使用的是STM32F103C8T6,这个库可以直接复制粘贴使用
我这是从OneNet平台上下载的一个8266库然后修改了一点东西,需要的可以直接去下载一份
#include "stm32f10x.h"
#include <esp8266.h>
#include "delay.h"
#include "usart.h"
#include <string.h>
#include <stdio.h>
#define ESP8266_WIFI_INFO "AT+CWJAP=\"232\",\"123456789\"\r\n"
#define ESP8266_ONENET_INFO "AT+CIPSTART=\"TCP\",\"116.62.81.138\",80\r\n"
unsigned char espbuf_Goal[300];//全局变量
unsigned char esp8266_buf[300];
unsigned short esp8266_cnt = 0, esp8266_cntPre = 0;
void ESP8266_Clear(void)//清除8266接收数据的包缓存
{
memset(esp8266_buf, 0, sizeof(esp8266_buf));
esp8266_cnt = 0;
}
_Bool ESP8266_WaitRecive(void)//这个不需要用到,可以不用管,复制就完了
{
if(esp8266_cnt == 0)
return REV_WAIT;
if(esp8266_cnt == esp8266_cntPre)
{
esp8266_cnt = 0;
return REV_OK;
}
esp8266_cntPre = esp8266_cnt;
return REV_WAIT;
}
_Bool ESP8266_SendCmd(char *cmd, char *res)//发送AT指令用的函数,复制就完了
{
unsigned char timeOut = 200;
Usart_SendString(USART1, (unsigned char *)cmd, strlen((const char *)cmd));
while(timeOut--)
{
if(ESP8266_WaitRecive() == REV_OK) //如果收到数据
{
if(strstr((const char *)esp8266_buf, res) != NULL) //如果检索到关键词
{
ESP8266_Clear(); //清空缓存
return 0;
}
}
delay_ms(10);
}
return 1;
}
//放延时可以给中断接收预留时间 该函数是返回GET请求获取的包数据
unsigned char *ESP8266_GetIPD(unsigned short timeOut)//这里我用了一个全局变量将esp8266buf储存到这个全局变量里面
{
do
{
delay_ms(5);
} while(timeOut--);
strcpy((char*)espbuf_Goal,(char*)esp8266_buf);
return espbuf_Goal;
}
//这个没什么好说的,就是初始化ESP8266的,要注意的是ESP8266的复位引脚要与PC14连接,否则复位指令失效
//直接复制就完了
void ESP8266_Init(void)
{
GPIO_InitTypeDef GPIO_Initure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
//ESP8266复位引脚
GPIO_Initure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_Initure.GPIO_Pin = GPIO_Pin_14; //GPIOC14-复位
GPIO_Initure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOC, &GPIO_Initure);
GPIO_WriteBit(GPIOC, GPIO_Pin_14, Bit_RESET);
delay_ms(250);
GPIO_WriteBit(GPIOC, GPIO_Pin_14, Bit_SET);
delay_ms(500);
ESP8266_Clear();
while(ESP8266_SendCmd("AT\r\n", "OK"))
delay_ms(500);
while(ESP8266_SendCmd("AT+CWMODE=1\r\n", "OK"))
delay_ms(500);
while(ESP8266_SendCmd("AT+RST\r\n", "OK"))
delay_ms(500);
while(ESP8266_SendCmd(ESP8266_WIFI_INFO, "GOT IP"))
delay_ms(500);
while(ESP8266_SendCmd("AT+CIPMUX=0\r\n", "OK"))
delay_ms(500);
while(ESP8266_SendCmd(ESP8266_ONENET_INFO, "CONNECT"))
delay_ms(500);
while(ESP8266_SendCmd("AT+CIPMODE=1\r\n", "OK"))
delay_ms(500);
while(ESP8266_SendCmd("AT+CIPSEND\r\n", ">"))
delay_ms(500);
UsartPrintf(USART2, "6. ESP8266 Init OK\r\n");//用来打印是否已经完成了配置
}
//我把USART1的中断往这个头文件写了,需要改的可以改一下
//里面的代码直接复制就完了
void USART1_IRQHandler(void)
{
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET) //接收中断
{
if(esp8266_cnt >= sizeof(esp8266_buf)) esp8266_cnt = 0; //防止串口被刷爆
esp8266_buf[esp8266_cnt++] = USART1->DR;//读取接收到的数据
USART_ClearFlag(USART1, USART_FLAG_RXNE);
}
}
头文件
#ifndef _ESP8266_H_
#define _ESP8266_H_
#define REV_OK 0 //接收完成标志
#define REV_WAIT 1 //接收未完成标志
unsigned char espbuf_Goal[300];
void ESP8266_Init(void);
void ESP8266_Clear(void);
void ESP8266_SendData(unsigned char *data, unsigned short len);
unsigned char *ESP8266_GetIPD(unsigned short timeOut);
#endif
2.单片机代码部分
main.c的代码
#include "stm32f10x.h"
#include "usart.h"
#include "string.h"
#include "delay.h"
#include "time.h"
#include "esp8266.h"
#include <stdio.h>
int main(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC|RCC_APB2Periph_GPIOB|RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOD| RCC_APB2Periph_AFIO,ENABLE);//串口时钟
delay_init();
Usart1_Init(115200);//物联网通信
Usart2_Init(115200);//打印串口信息
ESP8266_Init();//初始化物联网指令
ESP8266_Clear();//清除esp8266buf的缓存数据
char sss[256];
sprintf(sss,"GET https://api.seniverse/v3/weather/now.json?key=SOIDqYnBIHFXVAywS&location=zhaoqing&language=zh-Hans&unit=c\r\n");//发送get请求
Usart_SendString(USART1, (unsigned char *)sss, strlen((const char *)sss));
extern unsigned char espbuf_Goal[300];//调用esp8266.c的变量
ESP8266_GetIPD(200);//获取GET请求的返回包
ESP8266_Clear();//清除缓存数据
UsartPrintf(USART2, "%s\r\n",espbuf_Goal);
while(1)
{
}
}
可以看到我们这里已经成功收到来自心知天气返回的一个json的数据表包,而这个包已经储存在espbuf_Goal这个变量里面,剩下的就是如何对espbuf_Goal这个变量进行解析了,这个相信网上有很多资源跟教程,应该不难,或者等过几天再发一篇如何进行json解析这个数据包获取对应的数据
总结
由于是第一次发博客,所以有很多不好的地方需要改进,如果说还有看不懂不会用的可以多看几次多琢磨几次,先把以上的几个AT指令测试几次,就明白了,最后main.c那边的get请求!!! 千万记住最后要加\r\n 否则你是请求不出去的,至于上面还有一个usart的头文件,那个里面其实就是最基本的usar配置和中断优先级配置,这些你们可以自己去配置,也不难,可以多去看看一些其他博主的文章。
最后就是这个esp8266的代码我也不是原创,不过也不难,就是对usart_sendstring这个函数进行不断的封装还有串口中断接收数据而已,我只是提供一些适合初学者GET获取的简单方法,有其他什么不懂的可以留言,当然是对这篇文章的问题,其他太高深的问题我也不懂。以上代码有不好的地方可以指出,我也是初学者,写的可能不是很好。
更多推荐
stm32+esp8266 GET请求心知天气的简单方法
发布评论