STM32玩转物联网实战篇:2.ESP8266 WIFI模块TCP通信示例详解

1、准备开发板

开发板功能区分布图

在这里插入图片描述

开发板俯视图

在这里插入图片描述

2、ESP8266简介

    ESP8266 WIFI模块内置TCP/IP网络协议,模块支持三种网络模式,AP、STA和AP+STA模式,AP模式:模块作为WIFI热点,等待其他设备的连接,进行局域网的通信,STA模式:模块作为客户端通过路由器连接外网,和服务器进行通信,AP+STA模式:两种模式共存,可以进行任意切换。另外,模块支持AT指令操作,使用PC端或者单片机TTL串口配置简单的指令即可实现,这也是选择这款模块的一个原因。

什么是AT指令?

在这里插入图片描述

ESP8266开发常用的AT指令

基础AT指令
命令描述
AT测试AT启动
AT+RST重启模块
AT+GMR查看版本信息
WIFI功能AT指令
命令描述
AT+CWMODE选择WIFI应用模式
AT+CWJAP加入AP
AT+CWLAP列出当前可用AP
AT+CWQAP退出与AP的连接
AT+CWSAP设置AP模式下的参数
AT+CWLIF查看已接入设备的IP
TCP/IP 工具箱 AT 指令
命令描述
AT+CIPSTATUS获得连接状态
AT+CIPSTART建立TCP连接或注册UDP
AT+CIPSEND发送数据
AT+CIPCLOSE关闭TCP或UDP
AT+CIFSR获取本地IP地址
AT+CIPMUX启动多连接
AT+CIPSERVER配置为服务器
AT+CIPMODE设置模块传输模式
AT+CIPSTO设置服务器超时时间

3、在MDK中编写代码

    在写代码之前我们要了解的编程思维:高内聚、低耦合,简单来说就是:把一些东西放在一边,另一些东西放在另一边,然后划定边界,就是一种“分类”的思想,一个模块实现一个功能,功能之间相互联系。并且我们可以用I/O模型来管理我们的设备,在计算机系统中I/O就是输入(Input)和输出(Output)的意思,用来控制计算机的数据流动包括程序、硬件。

ESP8266基础函数
函数描述
ESP8266_IO_DelayESP8266模块延时
ESP8266_IO_Reset对ESP8266进行硬件重启
ESP8266_IO_SendESP8266模块所在的串口发送数据
ESP8266_IO_WaitRecive对ESP8266是否接收到数据进行判断
ESP8266_IO_ClearRecive清除ESP8266模块所在的串口的缓存
ESP8266功能函数
ESP8266_SendCmd对ESP8266发送命令,并将ESP8266返回的数据进行检索
ESP8266_SendDataESP8266向服务器发送数据
ESP8266_Net_Mode_Choose选择ESP8266模块的工作模式
ESP8266_JoinAPESP8266模块连接外部WiFi
ESP8266_CIPAP设置模块的 AP IP
ESP8266_BuildAPWF-ESP8266模块创建WiFi热点
ESP8266_Enable_MultipleIdESP8266模块启动多连接
ESP8266_Link_ServerESP8266模块连接外部服务器
ESP8266_StartOrShutServerESP8266模块开启或关闭服务器模式
ESP8266_Inquire_ApIp获取 F-ESP8266 的 AP IP
ESP8266_UnvarnishSend配置ESP8266模块进入透传发送
ESP8266_Get_LinkStatusESP8266 的连接状态,较适合单端口时使用
ESP8266_GetIPD对ESP8266返回的数据进行检索,并将处理好的数据返回

在ESP8266.h中编写以下代码

#ifndef __ESP8266_H_
#define __ESP8266_H_

#include "sys.h"

#define ESP8266_USART   LPUART1
#define USART_DEBUG 	USART1


#ifndef ESP8266_OK
#define ESP8266_OK                                              0
#endif

#ifndef ESP8266_NOK
#define ESP8266_NOK                                             1
#endif


#define  ESP8266_RETTYPE		unsigned char 

//ESP8266数据类型定义
typedef enum
{
	STA,
  	AP,
  	STA_AP  
}ENUM_Net_ModeTypeDef;

//ESP8266网络模式定义
typedef enum{
	 enumTCP,
	 enumUDP,
} ENUM_NetPro_TypeDef;

//ESP8266多连接ID定义
typedef enum{
	Multiple_ID_0 = 0,
	Multiple_ID_1 = 1,
	Multiple_ID_2 = 2,
	Multiple_ID_3 = 3,
	Multiple_ID_4 = 4,
	Single_ID_0 = 5,
} ENUM_ID_NO_TypeDef;

//ESP8266 AP加密方式定义
typedef enum{
	OPEN = 0,
	WEP = 1,
	WPA_PSK = 2,
	WPA2_PSK = 3,
	WPA_WPA2_PSK = 4,
} ENUM_AP_PsdMode_TypeDef;


#define User_ESP8266_BulitApSsid	  "PRECHIN"	  //要建立的热点的名称
#define User_ESP8266_BulitApEcn	  	  OPEN            //要建立的热点的加密方式
#define User_ESP8266_BulitApPwd  	  "prechin"      //要建立的热点的密钥

#define User_ESP8266_TCPServer_IP	  "192.168.1.119"	  //服务器开启的IP地址
#define User_ESP8266_TCPServer_PORT	  "8080"	  //服务器开启的端口

#define User_ESP8266_TCPServer_OverTime	  "1800"	  //服务器超时时间(单位:秒)

#define TCPAGREEMENT 	enumTCP	//TCP通信协议

typedef struct _NET_DEVICE_INFO
{

	USART_INFO_STRUCT* netIOInfo;	//串口句柄
	
	char staName[20];				//缓存SSID
	char staPass[20];				//缓存PassWord
	char staIPAddress[20];	//缓存IP
	char staPort[20];				//缓存Port
	
	unsigned short err : 2; 		//错误类型
	unsigned short netWork : 1;	//网络连接状态
	unsigned short initStep : 4;	//初始化步骤
	unsigned short dataType : 4;	//设定数据返回类型--16种
	unsigned short reverse : 6;		//预留

} NET_DEVICE_INFO;

extern NET_DEVICE_INFO netDeviceInfo;

void ESP8266_IO_Delay(uint32_t time);
void ESP8266_IO_Send(uint8_t *str,uint32_t strlen );
ESP8266_RETTYPE ESP8266_IO_WaitRecive(void);
void ESP8266_IO_ClearRecive(void);
void ESP8266_IO_Reset(void);

ESP8266_RETTYPE ESP8266_SendCmd(char *cmd, char *res1,char *res2,uint32_t timeOut);
ESP8266_RETTYPE ESP8266_SendData(FunctionalState enumEnUnvarnishTx,unsigned char *data, unsigned short len,ENUM_ID_NO_TypeDef ucId);
ESP8266_RETTYPE ESP8266_Net_Mode_Choose(ENUM_Net_ModeTypeDef enumMode);
ESP8266_RETTYPE ESP8266_JoinAP( char * pSSID, char * pPassWord );
ESP8266_RETTYPE ESP8266_CIPAP ( char * pApIp );
ESP8266_RETTYPE ESP8266_BuildAP ( char * pSSID, char * pPassWord, ENUM_AP_PsdMode_TypeDef enunPsdMode );
ESP8266_RETTYPE ESP8266_Enable_MultipleId (FunctionalState enumEnUnvarnishTx );
ESP8266_RETTYPE ESP8266_Link_Server(ENUM_NetPro_TypeDef enumE, char * ip, char * ComNum, ENUM_ID_NO_TypeDef id);
ESP8266_RETTYPE ESP8266_StartOrShutServer ( FunctionalState enumMode, char * pPortNum, char * pTimeOver );
ESP8266_RETTYPE ESP8266_Inquire_ApIp ( char * pApIp, uint8_t ucArrayLength );
ESP8266_RETTYPE ESP8266_UnvarnishSend ( void );
ESP8266_RETTYPE ESP8266_Get_LinkStatus ( void );
unsigned char *ESP8266_GetIPD(FunctionalState enumEnUnvarnishTx,unsigned short timeOut);
ESP8266_RETTYPE NET_DEVICE_Init(void);

#endif

在ESP8266.c编写以下代码


NET_DEVICE_INFO netDeviceInfo = {
																&lpuart1Info,		//ESP8266所在串口
																"M5GM",				//WIFI名称
																"18022100@MMKJ",	//WIFI密码
																"192.168.0.100",	//IP地址
																"7437",				//Port端口
																0, 0, 0, 0,	
																}; 

//ESP8266模块延时
void ESP8266_IO_Delay(uint32_t time)
{
	#if SYSTEM_SUPPORT_OS 
		rt_thread_mdelay(time);
	
	#else
		delay_ms(time);
	#endif
	
}

//ESP8266模块所在的串口发送数据
void ESP8266_IO_Send(uint8_t *str,uint32_t strlen )
{
	
	USART_SendBuf(ESP8266_USART,str,strlen);
	
}

//对ESP8266是否接收到数据进行判断
//1:接收失败 返回0:接收成功
ESP8266_RETTYPE ESP8266_IO_WaitRecive(void)
{

	if(netDeviceInfo.netIOInfo->InfBit.dataLen == 0) 						//如果接收计数为0 则说明没有处于接收数据中,所以直接跳出,结束函数
		return ESP8266_NOK;
		
	if(netDeviceInfo.netIOInfo->InfBit.dataLen == netDeviceInfo.netIOInfo->dataLenPre)	//如果上一次的值和这次相同,则说明接收完毕
	{
		netDeviceInfo.netIOInfo->rxBuf[netDeviceInfo.netIOInfo->InfBit.dataLen] = '\0';	//将数据的最后一位加上'\0'用于截断字符串,这样下一次接收就不用清除缓存了
		netDeviceInfo.netIOInfo->dataLenPre = netDeviceInfo.netIOInfo->InfBit.dataLen;	//将这一次的长度进行保存,当要检查接收长度时会用到
		
		netDeviceInfo.netIOInfo->InfBit.dataLen = 0;						//清0接收计数
			
		return ESP8266_OK;								//返回接收完成标志
	}
		
	netDeviceInfo.netIOInfo->dataLenPre = netDeviceInfo.netIOInfo->InfBit.dataLen;		//置为相同
	
	return ESP8266_NOK;								//返回接收未完成标志
}


//清除ESP8266模块所在的串口的缓存
void ESP8266_IO_ClearRecive(void)
{

	netDeviceInfo.netIOInfo->InfBit.dataLen = 0;
	
	memset(netDeviceInfo.netIOInfo->rxBuf, 0, sizeof(netDeviceInfo.netIOInfo->rxBuf));

}

//对ESP8266进行硬件重启(与AT指令重启不同,这里是控制ESP8266的RESET引脚强制重启)
void ESP8266_IO_Reset(void)
{
	
	UsartPrintf(USART_DEBUG, "Tips:	NET_DEVICE_Reset\r\n");
	
//	NET_DEVICE_RST_OFF;		//结束复位
//	ESP8266_IO_Delay(250);
//	
//	NET_DEVICE_RST_ON;		//复位
//	ESP8266_IO_Delay(500);
	
}

//对ESP8266发送命令,并将ESP8266返回的数据进行检索
// cmd:待发送的AT命令(注意要添加换行符)
// res1: 检索字段1
// res2: 检索字段2
// timeOut: 超时时间
//返回1:选择失败 0:选择成功
ESP8266_RETTYPE ESP8266_SendCmd(char *cmd, char *res1,char *res2,uint32_t timeOut)
{
	
	ESP8266_RETTYPE ucExecRes = ESP8266_NOK;  
	
	ESP8266_IO_ClearRecive();	//清除缓存
	
	UsartPrintf(USART_DEBUG,"Tips: %s\r\n",cmd);
	
	
	ESP8266_IO_Send((unsigned char *)cmd, strlen((const char *)cmd));	//写命令到网络设备
//	UsartPrintf(ESP8266_USART,"%s\r\n",cmd);
	
	while(timeOut--)												//等待
	{

		if(ESP8266_IO_WaitRecive() == ESP8266_OK)							//如果收到数据
		{
//		printf("\r\n=============start=================\r\n");
//		printf("netDeviceInfo.netIOInfo->rxBuf:%s\r\n",netDeviceInfo.netIOInfo->rxBuf);
//		printf("netDeviceInfo.netIOInfo->InfBit.dataLen:%d\r\n",netDeviceInfo.netIOInfo->InfBit.dataLen);
//		printf("\r\n===============end===============\r\n");
			if(*res1 != 0 && *res2 != 0)
			{
				if(((bool)strstr((const char *)netDeviceInfo.netIOInfo->rxBuf, res1))||
								((bool)strstr((const char *)netDeviceInfo.netIOInfo->rxBuf, res2)))
				{
						ucExecRes = ESP8266_OK;
						break;
				}
			}
			else if(*res1 != 0)
			{
				if((bool)strstr((const char *)netDeviceInfo.netIOInfo->rxBuf, res1))
				{
						ucExecRes = ESP8266_OK;
						break;
				}
			}
			else
			{
				if((bool)strstr((const char *)netDeviceInfo.netIOInfo->rxBuf, res2))	//如果检索到关键词
				{
						ucExecRes = ESP8266_OK;
						break;
				}
			}
		}
		ESP8266_IO_Delay(1);											//挂起等待
	}
	
//	ESP8266_IO_ClearRecive();								//清空缓存
	return ucExecRes;
}

//对ESP8266返回的数据进行检索,并将处理好的数据返回
// enumMode:工作模式
// timeOut: 超时时间
//返回1:选择失败 0:选择成功
unsigned char *ESP8266_GetIPD(FunctionalState enumEnUnvarnishTx,unsigned short timeOut)
{
//	unsigned char byte = 0, count = 0;
//	char sByte[5];
	char *ptrIPD;

	do
	{
		if(ESP8266_IO_WaitRecive() == ESP8266_OK)								//如果接收完成
		{
			if(enumEnUnvarnishTx == DISABLE)		//如果不是透传模式
			{
//					netDeviceInfo.netIOInfo->rxBuf[netDeviceInfo.netIOInfo->InfBit.dataLen++] = '\0';
//					printf("\r\n=============start=================\r\n");
//					printf("netDeviceInfo.netIOInfo->rxBuf:%s\r\n",netDeviceInfo.netIOInfo->rxBuf);
//					printf("netDeviceInfo.netIOInfo->InfBit.dataLen:%d\r\n",netDeviceInfo.netIOInfo->InfBit.dataLen);
//					printf("\r\n===============end===============\r\n");
				
					ptrIPD = strstr((char *)netDeviceInfo.netIOInfo->rxBuf, "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;
					}
			}
			else
			{
				return netDeviceInfo.netIOInfo->rxBuf;		//如果是透传模式直接返回
			}
		}
		ESP8266_IO_Delay(20);												//延时等待
	} while(timeOut--);

	return NULL;														//超时还未找到,返回空指针

}

//ESP8266向服务器发送数据
//返回1:发送失败 0:发送成功
ESP8266_RETTYPE ESP8266_SendData(FunctionalState enumEnUnvarnishTx,unsigned char *data, unsigned short len,ENUM_ID_NO_TypeDef ucId)
{
	
	char cmdBuf[30];
	ESP8266_RETTYPE ucExecRes = ESP8266_NOK;
	
	if( enumEnUnvarnishTx )	//如果是透传模式
	{
		
		ESP8266_IO_Send(data, len);	//直接发送数据
		ucExecRes = ESP8266_OK;
	}
	else
	{
		if ( ucId < 5 )		//如果是多连接
			sprintf ( cmdBuf, "AT+CIPSEND=%d,%d\r\n", ucId, len );	

		else			//如果是单连接
			sprintf ( cmdBuf, "AT+CIPSEND=%d\r\n", len );	
		
		ucExecRes = ESP8266_SendCmd ( cmdBuf, "> ", 0, 4000 );	
		if(ucExecRes == ESP8266_OK)
		{
			
			USART_SendBuf(USART_DEBUG,data,len);		//将数据在串口1打印出来
			
			ESP8266_IO_Send(data,len);	//将数据发送出去
			
		}
	}
	
	return ucExecRes;
	
}

//选择ESP8266模块的工作模式
// enumMode:工作模式
//返回1:选择失败 0:选择成功
ESP8266_RETTYPE ESP8266_Net_Mode_Choose(ENUM_Net_ModeTypeDef enumMode)
{
	switch ( enumMode )
	{
		case STA:
			return ESP8266_SendCmd ( "AT+CWMODE=1\r\n", "OK", "no change", 2500 ); 
		
	  	case AP:
		  	return ESP8266_SendCmd ( "AT+CWMODE=2\r\n", "OK", "no change", 2500 ); 
		
		case STA_AP:
		  	return ESP8266_SendCmd ( "AT+CWMODE=3\r\n", "OK", "no change", 2500 ); 
		
	  	default:
		  return ESP8266_NOK;
	}		
}

//ESP8266模块连接外部WiFi
//pSSID:WiFi名称字符串
//pPassWord:WiFi密码字符串
//返回1:连接失败 0:连接成功
ESP8266_RETTYPE ESP8266_JoinAP( char * pSSID, char * pPassWord )
{
	char cmdBuf [120];

	sprintf ( cmdBuf, "AT+CWJAP=\"%s\",\"%s\"\r\n", pSSID, pPassWord );
	
	return ESP8266_SendCmd( cmdBuf, "OK", NULL, 5000 );
	
}

/*
 * 函数名:ESP8266_CIPAP
 * 描述  :设置模块的 AP IP
 * 输入  :pApIp,模块的 AP IP
 * 返回  : 1,设置失败
 *         0,设置成功
 * 调用  :被外部调用
 */
ESP8266_RETTYPE ESP8266_CIPAP ( char * pApIp )
{
	char cmdBuf [ 30 ];	
	
	sprintf ( cmdBuf, "AT+CIPAP=\"%s\"\r\n", pApIp );
	
  if ( ESP8266_SendCmd ( cmdBuf, "OK", 0, 5000 ) == ESP8266_OK)
		return ESP8266_OK;
	else 
		return ESP8266_NOK;
	
}

/*
 * 函数名:ESP8266_BuildAP
 * 描述  :WF-ESP8266模块创建WiFi热点
 * 输入  :pSSID,WiFi名称字符串
 *       :pPassWord,WiFi密码字符串
 *       :enunPsdMode,WiFi加密方式代号字符串
 * 返回  : 1,创建失败
 *         0,创建成功
 * 调用  :被外部调用
 */
ESP8266_RETTYPE ESP8266_BuildAP ( char * pSSID, char * pPassWord, ENUM_AP_PsdMode_TypeDef enunPsdMode )
{
	char cmdBuf [120];

	sprintf ( cmdBuf, "AT+CWSAP=\"%s\",\"%s\",1,%d\r\n", pSSID, pPassWord, enunPsdMode );
	
	return ESP8266_SendCmd ( cmdBuf, "OK", 0, 1000 );
	
}

//ESP8266模块启动多连接
//enumEnUnvarnishTx:配置是否多连接
//返回1:配置失败 0:配置成功
ESP8266_RETTYPE ESP8266_Enable_MultipleId (FunctionalState enumEnUnvarnishTx )
{
	char cStr [20];
	
	sprintf ( cStr, "AT+CIPMUX=%d\r\n", ( enumEnUnvarnishTx ? 1 : 0 ) );
	
	return ESP8266_SendCmd ( cStr, "OK", 0, 500 );
	
}

//ESP8266模块连接外部服务器
//enumE:网络协议
//ip:服务器IP字符串
//ComNum:服务器端口字符串
//id:模块连接服务器的ID
//返回1:连接失败 0:连接成功
ESP8266_RETTYPE ESP8266_Link_Server(ENUM_NetPro_TypeDef enumE, char * ip, char * ComNum, ENUM_ID_NO_TypeDef id)
{
	char cStr [100] = { 0 }, cmdBuf [120];

  	switch (  enumE )
  	{
		case enumTCP:
		  sprintf ( cStr, "\"%s\",\"%s\",%s\r\n", "TCP", ip, ComNum );
		  break;
		
		case enumUDP:
		  sprintf ( cStr, "\"%s\",\"%s\",%s\r\n", "UDP", ip, ComNum );
		  break;
		
		default:
			break;
  	}

  	if ( id < 5 )
    	sprintf ( cmdBuf, "AT+CIPSTART=%d,%s\r\n", id, cStr);

  	else
	  	sprintf ( cmdBuf, "AT+CIPSTART=%s\r\n", cStr );

	return ESP8266_SendCmd ( cmdBuf, "OK", "ALREAY CONNECT", 4000 );
	
}

/*
 * ESP8266_DisconnectServer
 * 描述  :WF-ESP8266模块断开外部服务器
 * 输入  :id:模块断开服务器的ID
 * 
 * 返回  : 1,操作成功
 *         0,操作失败
 * 调用  :被外部调用
 */
ESP8266_RETTYPE ESP8266_DisconnectServer (ENUM_ID_NO_TypeDef id)
{
	char cmdBuf1 [120], cmdBuf2 [120];

	sprintf ( cmdBuf1, "AT+CLOSED=%d\r\n", id );
	
	return ( ESP8266_SendCmd ( cmdBuf1, "OK", "ERROR", 500 ) );
	
}


/*
 * 函数名:ESP8266_StartOrShutServer
 * 描述  :WF-ESP8266模块开启或关闭服务器模式
 * 输入  :enumMode,开启/关闭
 *       :pPortNum,服务器端口号字符串
 *       :pTimeOver,服务器超时时间字符串,单位:秒
 * 返回  : 1,操作失败
 *         0,操作成功
 * 调用  :被外部调用
 */
ESP8266_RETTYPE ESP8266_StartOrShutServer ( FunctionalState enumMode, char * pPortNum, char * pTimeOver )
{
	char cmdBuf1 [120], cmdBuf2 [120];

	if ( enumMode )
	{
		sprintf ( cmdBuf1, "AT+CIPSERVER=%d,%s\r\n", 1, pPortNum );
		
		sprintf ( cmdBuf2, "AT+CIPSTO=%s\r\n", pTimeOver );

		return ( ESP8266_SendCmd ( cmdBuf1, "OK", 0, 500 ) &&
						 ESP8266_SendCmd ( cmdBuf2, "OK", 0, 500 ) );
	}
	
	else
	{
		sprintf ( cmdBuf1, "AT+CIPSERVER=%d,%s\r\n", 0, pPortNum );

		return ESP8266_SendCmd ( cmdBuf1, "OK", 0, 500 );
	}
	
}

/*
 * 函数名:ESP8266_Inquire_ApIp
 * 描述  :获取 F-ESP8266 的 AP IP
 * 输入  :pApIp,存放 AP IP 的数组的首地址
 *         ucArrayLength,存放 AP IP 的数组的长度
 * 返回  : 0,获取成功
 *         1,获取失败
 * 调用  :被外部调用
 */
ESP8266_RETTYPE ESP8266_Inquire_ApIp ( char * pApIp, uint8_t ucArrayLength )
{
	char uc;
	
	char * pCh;
	
	
	if(!ESP8266_SendCmd ( "AT+CIFSR\r\n", "OK", 0, 500 ))
	{
		pCh = strstr ( (char*)netDeviceInfo.netIOInfo->rxBuf, "APIP,\"" );
		
		if ( pCh )
			pCh += 6;
		
		else
			return ESP8266_NOK;
		
		for ( uc = 0; uc < ucArrayLength; uc ++ )
		{
			pApIp [ uc ] = * ( pCh + uc);
			
			if ( pApIp [ uc ] == '\"' )
			{
				pApIp [ uc ] = '\0';
				break;
			}
		}
	}
	
	return ESP8266_OK;
	
}

//配置ESP8266模块进入透传发送
//返回1:配置失败 0:配置成功
ESP8266_RETTYPE ESP8266_UnvarnishSend ( void )
{
	if (!ESP8266_SendCmd ( "AT+CIPMODE=1\r\n", "OK", 0, 500 ))
		return ESP8266_NOK;
	
	return 
	  	ESP8266_SendCmd( "AT+CIPSEND\r\n", "OK", ">", 500 );
	
}

//ESP8266 的连接状态,较适合单端口时使用
//返回0:获取状态失败
//返回2:获得ip
//返回3:建立连接 
//返回4:失去连接 
ESP8266_RETTYPE ESP8266_Get_LinkStatus ( void )
{
	if (ESP8266_SendCmd( "AT+CIPSTATUS\r\n", "STATUS:", NULL, 1000 ) == ESP8266_OK)
	{
		UsartPrintf(USART_DEBUG,"netDeviceInfo.netIOInfo->rxBuf:%s\r\n",netDeviceInfo.netIOInfo->rxBuf);
		if ( strstr ( (char*)netDeviceInfo.netIOInfo->rxBuf, "STATUS:2\r\n" ) )
		{
				UsartPrintf(USART_DEBUG, "ESP8266 Got IP\r\n");
			return 2;
		}
		else if ( strstr ( (char*)netDeviceInfo.netIOInfo->rxBuf, "STATUS:3\r\n" ) )
		{
			UsartPrintf(USART_DEBUG, "ESP8266 Connect OK\r\n");
			return 3;
		}
		else if ( strstr ( (char*)netDeviceInfo.netIOInfo->rxBuf, "STATUS:4\r\n" ) )
		{
			UsartPrintf(USART_DEBUG, "ESP8266 Lost Connect\r\n");
			return 4;		
		}
	}
	UsartPrintf(USART_DEBUG, "ESP8266 TimeOut\r\n");			//获取状态失败
	return ESP8266_NOK;
	
}

//网络设备连接服务器前进行的初始化
//返回1:配置失败 0:配置成功
ESP8266_RETTYPE NET_DEVICE_LinkServer_Init(void)
{
	
	unsigned char errCount = 0, errType = 0;
//	char cfgBuffer[70];
	
	switch(netDeviceInfo.initStep)
	{
		case 0:
			if(ESP8266_Net_Mode_Choose(STA) == ESP8266_OK)
				netDeviceInfo.initStep++;
			break;
		case 1:
			if(ESP8266_Enable_MultipleId(DISABLE) == ESP8266_OK)
				netDeviceInfo.initStep++;
			break;
		case 2:
			if(ESP8266_JoinAP(netDeviceInfo.staName,netDeviceInfo.staPass) == ESP8266_OK)
				netDeviceInfo.initStep++;
			break;
		case 3:
			if(ESP8266_Link_Server(enumTCP,netDeviceInfo.staIPAddress,netDeviceInfo.staPort,Single_ID_0) == ESP8266_OK)
				netDeviceInfo.initStep++;
			break;
		default:
			 netDeviceInfo.netWork = 1;	
			 errType = 3;
			break;
	}
	
	
		return errType;
	
}

//网络设备初始化
//返回1:配置失败 0:配置成功
ESP8266_RETTYPE NET_DEVICE_Init(void)
{
	
	netDeviceInfo.err = NET_DEVICE_LinkServer_Init(); 									//获取连接结果	0-成功
	
	if(netDeviceInfo.err == 1) 												//如果路由信息错误,则启动ap模式 通过手机获取ssid和password
	{
		UsartPrintf(USART_DEBUG, "Wifi info Error,Use USART2 -> 8266\r\n");
        
    return ESP8266_NOK;
	}
	else if(netDeviceInfo.err == 2) 										//如果是平台信息错误
	{
		UsartPrintf(USART_DEBUG, "PT info Error,Use APP -> 8266\r\n");
		
		return ESP8266_NOK;
	}
	else if(netDeviceInfo.err == 3)
	{
		UsartPrintf(USART_DEBUG, "Tips:	NET_DEVICE STA OK\r\n");
		
		ESP8266_IO_Delay(500); //延时提示
		
		return ESP8266_OK;
	}
	else
		return ESP8266_NOK;

}

在main.c中添加以下代码

/* USER CODE BEGIN PTD */
ESP8266_RETTYPE netStatus = ESP8266_NOK;
unsigned char netErrCount = 0;
/* USER CODE END PTD */

在main.c下的main函数加入以下代码

  /* USER CODE BEGIN 1 */
	unsigned char* dataPtr = NULL;
	uint32_t send_time = 0;
	ESP8266_RETTYPE ucExecRes = ESP8266_NOK;

  /* USER CODE END 1 */
 /* USER CODE BEGIN 2 */
	
	USART_Interupt_Enable();		//使能串口中断
	TIM_Interupt_Enable();			//使能定时器中断

  /* USER CODE END 2 */
/* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
		
		if(!netDeviceInfo.netWork)	如果网络未连接
		{
			if(NET_DEVICE_Init() == ESP8266_OK)
			{
				printf("连接服务器成功\r\n");
			}
		}
		if(netDeviceInfo.netWork)	//如果网络连接成功
		{
			dataPtr = ESP8266_GetIPD(DISABLE,0);	//解析服务器下发的数据
			if(dataPtr != NULL)
			{
				printf("dataPtr:%s\r\n",dataPtr);
				if(strstr((char*)dataPtr,"OpenLED"))
				{
					printf("LED点亮\r\n");
					LED_Set(LED_ON);
				}
				if(strstr((char*)dataPtr,"CloseLED"))
				{
					printf("LED熄灭\r\n");
					LED_Set(LED_OFF);
				}
			}
		}
		
		if(time2Count - send_time >= 10000)		//(1ms * 2000)相当于延时2秒钟
		{
			send_time = time2Count;				//记下当前定时器的数值
			if(netDeviceInfo.netWork)	//如果网络连接成功
			{
				ucExecRes = ESP8266_SendData(DISABLE,"hello world",strlen("hello world"),Single_ID_0);	//向服务器发送数据
				if(ucExecRes != ESP8266_OK)
				{
					printf("网络可能断开了\r\n");
					netErrCount++;	//错误次数进行累加
					if(netErrCount >= 3)	//超过三次,进行自检
					{
						netStatus = ESP8266_Get_LinkStatus();	//检查连接状态
						if(netStatus == 4)		//网络已经断开
						{
								netErrCount = 0;		//将错误清零
								netDeviceInfo.netWork = 0;	//标志网络断开
								netDeviceInfo.initStep = 0;	//将初始化步骤清零
						}
					}
				}
				else
				{
					netErrCount = 0;	//无错误,清除错误计数
				}
			}
		}	
  }
  /* USER CODE END 3 */

4、实验现象

实现的功能
1、上电自动连接WIFI
2、成功连接WIFI后自动连接TCP服务器
3、成功连接TCP服务器后自动发送数据给服务器
4、TCP服务器下发OpenLED指令会点亮LED,下发CloseLED会熄灭LED灯
5、当服务器连接断开的时候,会自动进行重新连接服务器,并重新发送数据

在这里插入图片描述

  • 38
    点赞
  • 361
    收藏
    觉得还不错? 一键收藏
  • 11
    评论
WLAN是英文WirelessLAN的缩写,就是无线局域网的意思。无线以太网技术是一种基于无线传输的局域网技术,与有线网络技术相比,具有灵活、建网迅速、个人化等特点。将这一技术应用于电信网的接入网领域,能够方便、灵活地为用户提供网络接入,适合于用户流动性较大、有数据业务需求的公共场所、高端的企业及家庭用户、需要临时建网的场合以及难以采用有线接入方式的环境等 作为全球公认的局域网权威,IEEE802工作组建立的标准在过去二十年内在局域网领域独领风骚。这些协议包括了802.3Ethernet协议、802.5TokenRing协议、802.3z100BASE-T快速以太网协议。在1997年,经过了7年的工作以后,IEEE发布了802.11协议,这也是在无线局域网领域内的第一个国际上被认可的协议。  在1999年9月,他们又提出了802.11b"HighRate"协议,用来对802.11协议进行补充,802.11b在802.11的1Mbps和2Mbps速率下又增加了5.5Mbps和11Mbps两个新的网络吞吐速率。利用802.11b,移动用户能够获得同Ethernet一样的性能、网络吞吐率、可用性。这个基于标准的技术使得管理员可以根据环境选择合适的局域网技术来构造自己的网络,满足他们的商业用户和其他用户的需求。802.11协议主要工作在ISO协议的最低两层上,并在物理层上进行了一些改动,加入了高速数字传输的特性和连接的稳定性。
### 回答1: ESP8266 WiFi 模块在 Arduino 中的一个常见例子是使用它来作为一个网络客户端,连接到一个网络服务器以发送和接收数据。下面是一个简单的示例代码: ``` #include <ESP8266WiFi.h> const char* ssid = "your_SSID"; const char* password = "your_PASSWORD"; const char* host = "your_host_or_IP_address"; void setup() { Serial.begin(115200); WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(1000); Serial.println("Connecting to WiFi..."); } Serial.println("Connected to WiFi"); } void loop() { WiFiClient client; if (!client.connect(host, 80)) { Serial.println("Connection failed"); return; } client.print("GET / HTTP/1.1\r\nHost: "); client.print(host); client.println("\r\n"); while (client.connected()) { if (client.available()) { char c = client.read(); Serial.write(c); } } client.stop(); Serial.println("Client disconnected"); delay(5000); } ``` 该代码将连接到指定的 WiFi 网络,然后创建一个客户端连接到指定的主机。当客户端连接时,代码将发送一个 HTTP GET 请求,并在接收到响应时打印响应内容。 ### 回答2: ESP8266 WIFI模块是一种便于接入WIFI的无线模块,极大简化了嵌入式系统设计者对于WIFI网络的接入。它包括一个32位的处理器,内置WIFI电路,集成TCP/IP协议栈以及其他有用的硬件以及软件资源。同时,它还支持透明串口以及AT指令,可以很方便地通过UART口与MCU(单片机)直接对接。Arduino作为下位机类型使用则可以更方便地实现联网控制。 接下来介绍一个基于ESP8266 WIFI模块与Arduino的实例。 开发板选型:NodeMCU ESP8266 ESP-12E 开发板 / WEMOS D1 mini Arduino IDE 这里以NodeMCU ESP8266 ESP-12E 开发板为例介绍如何使用ESP8266实现WIFI联网。 1. 准备工作 连接电脑和NodeMCU ESP8266 ESP-12E 开发板,执行以下步骤: · 安装Arduino IDE · 安装ESP8266开发板支持包 · 打开Arduino IDE,打开文件 -> 首选项,将下面的URL复制到附加开发板管理器网址指示器文本框中:http://arduino.esp8266.com/stable/package_esp8266com_index.json · 打开工具菜单:工具 -> 开发板 -> 开发板管理器 -> 搜索esp8266 -> 安装 · 选择开发板:工具 -> 开发板 -> NodeMCU 1.0(ESP-12E Module) 2. 创建项目 依次选择文件 -> 新建,打开一个新的Sketch,复制以下源代码: #include <ESP8266WiFi.h> const char* ssid = "your_SSID_here"; // 你的WIFI名称 const char* password = "your_PASSWORD_here"; // 你的WIFI密码 void setup() { Serial.begin(9600); // 初始化串口 WiFi.begin(ssid, password); // 连接WIFI Serial.println("Connecting to WiFi"); while (WiFi.status() != WL_CONNECTED ) { // 等待WIFI连接 delay(1000); Serial.println("Connecting..."); } Serial.println("Connected"); } void loop() { } 其中ssid和password分别填入你的WIFI的名称和密码。 3. 调试项目 将NodeMCU ESP8266 ESP-12E 开发板通过数据线连接到电脑,并通过Arduino IDE将项目上传至开发板。 开发板连接至WIFI后,串口会输出“Connecting to WiFi”,然后每隔一秒钟输出“Connecting&hellip;”,一旦连接成功就会输出“Connected”。 总结: 通过上述实验可以清晰了解如何使用ESP8266 WIFI模块与NodeMCU ESP8266 ESP-12E 开发板结合,更好地应用在WIFI联网场景中。在工程项目中,我们可以通过调整源代码以及使用不同的WIFI模块,开发出符合应用场景需要的联网控制系统。 ### 回答3: ESP8266 WIFI模块是一种非常常见的物联网设备,它可以实现设备与网络的通信,与其它设备进行数据交互。我们可以使用Arduino开发板来控制和传输数据。 ESP8266模块使用AT指令与Arduino通信,可以在Arduino IDE中使用Serial Monitor进行控制。同时,我们还需要安装ESP8266的库和相应的驱动程序。安装完成后,我们可以进行一些简单的实验。 首先,我们需要连接ESP8266模块到Arduino开发板,将模块的TXD引脚连接到Arduino的RXD引脚,模块的RXD引脚连接到Arduino的TXD引脚。然后,将模块的VCC引脚连接到Arduino的+5V引脚,模块的GND引脚连接到GND引脚。 接下来,我们可以通过AT指令让ESP8266模块连接到WIFI网络。我们需要设置WIFI的SSID和密码,同时使ESP8266模块工作在STA(Station)模式。我们可以使用下面的代码来连接WIFI: AT+CWMODE=1 //设置模式为STA,即客户端模式 AT+CWJAP="SSID","password" //连接到WIFI网络,替换SSID和password为真实的网络名称和密码 连接成功后,我们可以通过AT指令获取到IP地址。我们可以使用下面的代码来获取IP地址: AT+CIFSR //获取IP地址 获取到IP地址后,我们就可以发送和接收数据了。我们可以使用AT+CWLAP指令来扫描WIFI网络信号。我们可以使用下面的代码来扫描网络: AT+CWLAP //扫描可用的WIFI网络 我们还可以使用AT+CWQAP指令来断开与WIFI网络的连接。我们可以使用下面的代码来断开连接: AT+CWQAP //断开WIFI连接 除了使用AT指令进行控制之外,我们还可以使用Arduino代码进行控制。可以使用ESP8266WiFi库连接到WIFI网络,发送数据和接收数据。我们可以使用下面的代码来连接WIFI: #include <ESP8266WiFi.h> const char* ssid = "yourNetworkSSID"; const char* password = "yourNetworkPassword"; void setup() { Serial.begin(115200); delay(10); // We start by connecting to a WiFi network Serial.println(); Serial.println(); Serial.print("Connecting to "); Serial.println(ssid); WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } Serial.println(""); Serial.println("WiFi connected"); Serial.println("IP address: "); Serial.println(WiFi.localIP()); } void loop() { // put your main code here, to run repeatedly: } 从上面的代码可以看到,我们首先设置WIFI的SSID和密码,然后连接到WIFI网络,并获取IP地址。我们可以使用WiFi.localIP()来获取IP地址。连接成功之后,我们就可以在loop函数中进行数据传输。我们可以使用WiFiClient类来连接到特定的IP地址和端口号,并发送数据。我们可以使用下面的代码来发送数据: #include <ESP8266WiFi.h> const char* ssid = "yourNetworkSSID"; const char* password = "yourNetworkPassword"; const char* host = "yourServerIP"; const int port = 80; void setup() { Serial.begin(115200); delay(10); // We start by connecting to a WiFi network Serial.println(); Serial.println(); Serial.print("Connecting to "); Serial.println(ssid); WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } Serial.println(""); Serial.println("WiFi connected"); Serial.println("IP address: "); Serial.println(WiFi.localIP()); } void loop() { // put your main code here, to run repeatedly: WiFiClient client; if (!client.connect(host, port)) { Serial.println("connection failed"); return; } client.print("data to send"); delay(5000); client.stop(); } 从上面的代码可以看到,我们可以使用WiFiClient类连接到特定的IP地址和端口号,并发送数据。我们还可以使用WiFiServer类作为服务器接收数据。我们可以使用下面的代码来接收数据: #include <ESP8266WiFi.h> const char* ssid = "yourNetworkSSID"; const char* password = "yourNetworkPassword"; WiFiServer server(80); void setup() { Serial.begin(115200); delay(10); // We start by connecting to a WiFi network Serial.println(); Serial.println(); Serial.print("Connecting to "); Serial.println(ssid); WiFi.begin(ssid, password); while (WiFi.status() != WL_CONNECTED) { delay(500); Serial.print("."); } Serial.println(""); Serial.println("WiFi connected"); Serial.println("IP address: "); Serial.println(WiFi.localIP()); server.begin(); } void loop() { // put your main code here, to run repeatedly: WiFiClient client = server.available(); if (client) { Serial.println("new client"); String currentLine = ""; while (client.connected()) { if (client.available()) { char c = client.read(); Serial.write(c); if (c == '\n') { if (currentLine.length() == 0) { client.println("HTTP/1.1 200 OK"); client.println("Content-type:text/html"); client.println(); client.print("Hello World!"); break; } else { currentLine = ""; } } else if (c != '\r') { currentLine += c; } } } client.stop(); } } 从上面的代码可以看到,我们可以使用WiFiServer类作为服务器接收数据,并通过连接到的客户端进行数据交互。 通过上述这些代码,我们可以使用ESP8266 WIFI模块实现了一些简单的数据传输,并将其与Arduino进行了结合,实现了物联网的应用。同时,我们也看到了ESP8266模块的一些基本原理和应用方法。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值