ESP-01S + STM32物联网

一、硬件结构

采用STM32系统板作为主控,ESP-01S作为WIFI模块,DHT温湿度传感器、0.96寸OLED屏。

二、串口驱动ESP8266

驱动ESP8266程序:

/**
	************************************************************
	************************************************************
	************************************************************
	*	文件名: 	esp8266.c
	*
	*	作者: 		张继瑞
	*
	*	日期: 		2017-05-08
	*
	*	版本: 		V1.0
	*
	*	说明: 		ESP8266的简单驱动
	*
	*	修改记录:	
	************************************************************
	************************************************************
	************************************************************
**/

//单片机头文件
#include "stm32f10x.h"

//网络设备驱动
#include "esp8266.h"

//硬件驱动
#include "delay.h"
#include "usart.h"

//C库
#include <string.h>
#include <stdio.h>


#define ESP8266_WIFI_INFO		"AT+CWJAP=\"2\",\"12345678\"\r\n"


unsigned char esp8266_buf[512];
unsigned short esp8266_cnt = 0, esp8266_cntPre = 0;


//==========================================================
//	函数名称:	ESP8266_Clear
//
//	函数功能:	清空缓存
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void ESP8266_Clear(void)
{

	memset(esp8266_buf, 0, sizeof(esp8266_buf));
	esp8266_cnt = 0;

}

//==========================================================
//	函数名称:	ESP8266_WaitRecive
//
//	函数功能:	等待接收完成
//
//	入口参数:	无
//
//	返回参数:	REV_OK-接收完成		REV_WAIT-接收超时未完成
//
//	说明:		循环调用检测是否接收完成
//==========================================================
_Bool ESP8266_WaitRecive(void)
{

	if(esp8266_cnt == 0) 							//如果接收计数为0 则说明没有处于接收数据中,所以直接跳出,结束函数
		return REV_WAIT;
		
	if(esp8266_cnt == esp8266_cntPre)				//如果上一次的值和这次相同,则说明接收完毕
	{
		esp8266_cnt = 0;							//清0接收计数
			
		return REV_OK;								//返回接收完成标志
	}
		
	esp8266_cntPre = esp8266_cnt;					//置为相同
	
	return REV_WAIT;								//返回接收未完成标志

}

//==========================================================
//	函数名称:	ESP8266_SendCmd
//
//	函数功能:	发送命令
//
//	入口参数:	cmd:命令
//				res:需要检查的返回指令
//
//	返回参数:	0-成功	1-失败
//
//	说明:		
//==========================================================
_Bool ESP8266_SendCmd(char *cmd, char *res)
{
	
	unsigned char timeOut = 200;

	Usart_SendString(USART2, (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;
			}
		}
		
		DelayXms(10);
	}
	
	return 1;

}

//==========================================================
//	函数名称:	ESP8266_SendData
//
//	函数功能:	发送数据
//
//	入口参数:	data:数据
//				len:长度
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void ESP8266_SendData(unsigned char *data, unsigned short len)
{

	char cmdBuf[32];
	
	ESP8266_Clear();								//清空接收缓存
	sprintf(cmdBuf, "AT+CIPSEND=%d\r\n", len);		//发送命令
	if(!ESP8266_SendCmd(cmdBuf, ">"))				//收到‘>’时可以发送数据
	{
		Usart_SendString(USART2, data, len);		//发送设备连接请求数据
	}

}

//==========================================================
//	函数名称:	ESP8266_GetIPD
//
//	函数功能:	获取平台返回的数据
//
//	入口参数:	等待的时间(乘以10ms)
//
//	返回参数:	平台返回的原始数据
//
//	说明:		不同网络设备返回的格式不同,需要去调试
//				如ESP8266的返回格式为	"+IPD,x:yyy"	x代表数据长度,yyy是数据内容
//==========================================================
unsigned char *ESP8266_GetIPD(unsigned short timeOut)
{

	char *ptrIPD = NULL;
	
	do
	{
		if(ESP8266_WaitRecive() == REV_OK)								//如果接收完成
		{
			ptrIPD = strstr((char *)esp8266_buf, "IPD,");				//搜索“IPD”头
			if(ptrIPD == NULL)											//如果没找到,可能是IPD头的延迟,还是需要等待一会,但不会超过设定的时间
			{
				//UsartPrintf(USART_DEBUG, "\"IPD\" not found\r\n");
			}
			else
			{
				ptrIPD = strchr(ptrIPD, ':');							//找到':'
				if(ptrIPD != NULL)
				{
					ptrIPD++;
					return (unsigned char *)(ptrIPD);
				}
				else
					return NULL;
				
			}
		}
		
		DelayXms(5);													//延时等待
	} while(timeOut--);
	
	return NULL;														//超时还未找到,返回空指针

}

//==========================================================
//	函数名称:	ESP8266_Init
//
//	函数功能:	初始化ESP8266
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
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);
//	DelayXms(250);
//	GPIO_WriteBit(GPIOC, GPIO_Pin_14, Bit_SET);
//	DelayXms(500);
	
	ESP8266_Clear();
	
	UsartPrintf(USART_DEBUG, "1. AT\r\n");//判断8266是否正常工作
	while(ESP8266_SendCmd("AT\r\n", "OK"))
		DelayXms(500);
	
	UsartPrintf(USART_DEBUG, "2. CWMODE\r\n");
	while(ESP8266_SendCmd("AT+CWMODE=1\r\n", "OK"))  //配置工作模式
		DelayXms(500);
	
	UsartPrintf(USART_DEBUG, "3. AT+CWDHCP\r\n");
	while(ESP8266_SendCmd("AT+CWDHCP=1,1\r\n", "OK"))
		DelayXms(500);
	
	UsartPrintf(USART_DEBUG, "4. CWJAP\r\n");
	while(ESP8266_SendCmd(ESP8266_WIFI_INFO, "GOT IP"))
		DelayXms(500);
	
	UsartPrintf(USART_DEBUG, "5. ESP8266 Init OK\r\n");

}

//==========================================================
//	函数名称:	USART2_IRQHandler
//
//	函数功能:	串口2收发中断
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void USART2_IRQHandler(void)
{

	if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET) //接收中断
	{
		if(esp8266_cnt >= sizeof(esp8266_buf))	esp8266_cnt = 0; //防止串口被刷爆
		esp8266_buf[esp8266_cnt++] = USART2->DR;
		
		USART_ClearFlag(USART2, USART_FLAG_RXNE);
	}

}

其中以下函数是发送命令的函数

_Bool ESP8266_SendCmd(char *cmd, char *res)
{
	
	unsigned char timeOut = 200;

	Usart_SendString(USART2, (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;
			}
		}
		
		DelayXms(10);
	}
	
	return 1;

}

 发送如下指令

1、问候一下ESP8266是否正常,如果正常return 0,跳出while循环,进行第二步。

	UsartPrintf(USART_DEBUG, "1. AT\r\n");//判断8266是否正常工作
	while(ESP8266_SendCmd("AT\r\n", "OK"))
		DelayXms(500);

2、告诉ESP8266等下要连接一个WIFI,让它做好准备,如果准备好了,进行第三步。

    UsartPrintf(USART_DEBUG, "2. CWMODE\r\n");
    while(ESP8266_SendCmd("AT+CWMODE=1\r\n", "OK"))  //配置工作模式
        DelayXms(500);

3、告诉ESP8266去获取IP地址,如果获取成功了,就进行第四步。

	UsartPrintf(USART_DEBUG, "3. AT+CWDHCP\r\n");	
    while(ESP8266_SendCmd("AT+CWDHCP=1,1\r\n", "OK"))
		DelayXms(500);

4、告诉ESP8266可以去连接WIFI了,告诉它一个具体的WIFI名称和WIFI密码,如果连接成功,打印ESP8266初始化成功。

	UsartPrintf(USART_DEBUG, "4. CWJAP\r\n");
	while(ESP8266_SendCmd(ESP8266_WIFI_INFO, "GOT IP"))
		DelayXms(500);

 其中发送指令的介绍如下:

参考博客:无线WIFI模块esp8266的AT指令操作_8266关闭dhcp-CSDN博客

 AT+CWMODE=x       设置模式为x,1:station模式  2:ap模式  3:两个模式同时

AT+CWDHCP=x,y     开启dhcp,y=0关闭,1开启,x为0时是ap,1是station, 2是二者同时

AT+CWJAP="xxx","yyy"    当作为station模式时,加入热点xxx,xxx是热点SSID,yyy是热点密码(和手机连接wifi类似)

#define ESP8266_WIFI_INFO		"AT+CWJAP=\"2\",\"12345678\"\r\n"

//第一个是WIFI名称,第二个是WIFI密码

dhcp是一种局域网IP管理协议,用于动态分配IP。ESP8266开启dhcp后,它会向路由器发起dhcp请求,然后路由器就会给它分配一个IP地址

三、连接MQTT服务器

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: ESP8266-01SSTM32的接线方式有多种,具体取决于您的应用场景和需求。以下是一种常见的接线方式: 1. 将ESP8266-01S的VCC引脚连接到STM32的3.3V电源引脚上。 2. 将ESP8266-01S的GND引脚连接到STM32的GND引脚上。 3. 将ESP8266-01S的TXD引脚连接到STM32的RX引脚上。 4. 将ESP8266-01S的RXD引脚连接到STM32的TX引脚上。 5. 如果需要使用ESP8266-01S的GPIO引脚,可以将其连接到STM32的任意GPIO引脚上。 需要注意的是,ESP8266-01S的工作电压为3.3V,不能直接连接到STM32的5V引脚上,否则可能会损坏ESP8266-01S。同时,ESP8266-01S的TXD和RXD引脚需要通过电平转换器进行转换,以适应STM32的3.3V电平。 ### 回答2: ESP8266-01S是一款非常实用的无线模块,可与STM32芯片相连接。下面将会介绍如何与STM32正确地接线。 首先,ESP8266-01S的接口分为VCC、GND、CH_PD、RX 和TX五个针脚。而STM32的IO口可以有多个,因此需要确认在STM32中可用的IO口,需要从STM32芯片的手册中查看。 接线步骤如下: 1. 将ESP8266-01S的TX接口连接到STM32芯片的RX端口。同时将ESP8266-01S的RX端口连接到STM32的TX端口。 2. 将ESP8266-01S的GND和CH_PD连接到STM32的GND端口。 3. ESP8266-01S的VCC端口需要通过3.3V电源供电,STM32的3.3V端口可以使用此端口直接供电。 4. 在STM32芯片中需要开启串口,并且选择与ESP8266-01S相匹配的参数,例如波特率和数据位、校验位以及停止位等一系列参数。 5. 如果需要在STM32上控制ESP8266-01S,可以在按下按钮或者开关时,通过STM32的IO口给ESP8266-01S的CH_PD针脚传递高电平信号,即可实现对ESP8266-01S的控制。 需要注意的是,在接线之前,必须保证STM32的IO口与ESP8266-01S的RX、TX信号完全对接。如果接口数量不够,也可以通过串口扩展板或者其他拓展方式来实现ESP8266-01S的连接。 总之,ESP8266-01S接线非常简单,但是需要注意每一个细节,确保接线稳定,这样才能实现准确高效的无线通讯。 ### 回答3: ESP8266-01S是一款内置WiFi模块的微控制器,而STM32则是一款性能强劲的微控制器。将它们连接起来可以实现许多有趣的物联网(IoT)应用。 ESP8266-01SSTM32的连接方式有两种:串口连接和SPI连接。 串口连接需要将ESP8266-01S的TX、RX引脚分别连接到STM32的TX、RX引脚。此外,还需要连接ESP8266-01S的VCC、GND引脚到STM32的3.3V和GND引脚。接线比较简单,但是需要配置串口通信的波特率和数据位、校验位、停止位等参数。 SPI连接需要连接ESP8266-01S的MOSI、MISO、CLK和CS引脚到STM32的相应引脚。此外,还需要连接ESP8266-01S的VCC、GND引脚到STM32的3.3V和GND引脚。SPI连接相对于串口连接稍微复杂一些,但是可以实现更加快速、可靠的数据传输。 ESP8266-01SSTM32的连接方式取决于具体应用场景和要求。如果应用场景要求低功耗、简单,则串口连接是不错的选择;如果应用场景要求高速、可靠,则SPI连接是更好的选择。 在使用ESP8266-01SSTM32进行开发时,需要掌握两个设备的相关知识,包括硬件结构、引脚功能等,并且需要掌握通信协议和相应的编程技术。同时,还要持续关注更新的技术和应用,以便优化系统设计和提高开发效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值