ESP8266 -- STM32与阿里云物联网平台建立通信(四)

目录

链接快速定位

前沿

1 准备工作

2 硬件环境介绍

3 软件环境介绍

3.1 串口初始化及配置

3.2 编写AT MQTT指令代码

3.2.1 ESP8266_Cmd函数介绍

3.2.2 wifi连接函数介绍

3.2.3 云端连接语句介绍

3.2.4 环回消息测试语句介绍

3.2.5 属性上报语句介绍

3.2.6 设置属性语句介绍

4 实验现象

4.1 代码运行现象

4.2 属性上报现象  

 4.3 设置属性现象


链接快速定位

ESP8266 -- 烧录AT固件(一)

ESP8266 -- 搭建阿里云物联网与MQTT.fx通信的平台(二)

ESP8266 -- 串口助手 AT MQTT 指令与阿里云物联网平台建立通信(三)

演示代码在文章顶部可以下载,也可以通过评论留下邮箱进行发送

前沿

        前面讲解了固件的烧录,搭建阿里云物联网平台和esp8266利用串口助手与阿里云物联网平台建立通信,本小节主要讲解利用STM32与esp8266模块进行通信。

1 准备工作

        在进行本小节内容的时候,需要做如下的准备工作:

  • 烧录AT MQTT固件,见链接:ESP8266 -- 烧录AT固件(一)
  • 阿里物联网平台的搭建,见链接:ESP8266 -- 搭建阿里云物联网与MQTT.fx通信的平台(二)

        如果对AT指令不是很熟悉,建议参考:ESP8266 -- 串口助手 AT MQTT 指令与阿里云物联网平台建立通信(三),简单的掌握一下AT指令。

2 硬件环境介绍

       硬件选择安信可的ESP01S模块与STM32单片机相连接,具体连接方式这里不展开讲解,网上也有很多资料,原理就是串口间的通信。

3 软件环境介绍

        软件环境使用keil5,编译下载stm32f103标准库程序进行演示。演示代码在文章顶部可以下载,也可以通过评论留下邮箱进行发送。下面重点讲解软件代码。

3.1 串口初始化及配置

         主要实现了串口的初始化配置,并且使能了接收完成中断。

static void ESP8266_USART_Config ( void )
{
	GPIO_InitTypeDef GPIO_InitStructure;
	USART_InitTypeDef USART_InitStructure;
	
	
	/* config USART clock */
	macESP8266_USART_APBxClock_FUN ( macESP8266_USART_CLK, ENABLE );
	macESP8266_USART_GPIO_APBxClock_FUN ( macESP8266_USART_GPIO_CLK, ENABLE );
	
	/* USART GPIO config */
	/* Configure USART Tx as alternate function push-pull */
	GPIO_InitStructure.GPIO_Pin =  macESP8266_USART_TX_PIN;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(macESP8266_USART_TX_PORT, &GPIO_InitStructure);  
  
	/* Configure USART Rx as input floating */
	GPIO_InitStructure.GPIO_Pin = macESP8266_USART_RX_PIN;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
	GPIO_Init(macESP8266_USART_RX_PORT, &GPIO_InitStructure);
	
	/* USART mode config */
	USART_InitStructure.USART_BaudRate = macESP8266_USART_BAUD_RATE;
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;
	USART_InitStructure.USART_StopBits = USART_StopBits_1;
	USART_InitStructure.USART_Parity = USART_Parity_No ;
	USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
	USART_Init(macESP8266_USARTx, &USART_InitStructure);

	/* 中断配置 */
	USART_ITConfig ( macESP8266_USARTx, USART_IT_RXNE, ENABLE ); //使能串口接收中断 

	ESP8266_USART_NVIC_Configuration ();
	
	USART_Cmd(macESP8266_USARTx, ENABLE);
}
void macESP8266_USART_INT_FUN ( void )
{
    uint8_t ucCh;

    if ( USART_GetITStatus ( macESP8266_USARTx, USART_IT_RXNE ) != RESET )
    {
        ucCh  = USART_ReceiveData( macESP8266_USARTx );

        if ( strEsp8266_Fram_Record .InfBit .FramLength < ( RX_BUF_MAX_LEN - 1 ) ) //预留1个字节写结束符
        {
            strEsp8266_Fram_Record .Data_RX_BUF [ strEsp8266_Fram_Record .InfBit .FramLength ++ ]  = ucCh;
        }
    }
}

3.2 编写AT MQTT指令代码

        以下就是AT MQTT实现的主要函数,此函数主要实现了以下功能:

  • wifi连接
  • 云端连接
  • 环回消息的测试
  • 属性上报
  • 设置属性
void ESP8266_Test(void)
{
	char cCmd [254];  //最多一次发送256字节,除去两个字节的换行符

    printf ( "正在配置 ESP8266 ......\r\n" );

    macESP8266_CH_ENABLE();
    
    ESP8266_AT_Test ();

    ESP8266_Net_Mode_Choose (STA);

    while ( ! ESP8266_JoinAP (ESP8266_APSSId, ESP8266_APPWD) );  
    printf("wifi已连接\r\n");
    
    /* 单连接模式 */
    ESP8266_Enable_MultipleId ( DISABLE );
    
    /* 配置云端参数 */
    sprintf (cCmd, "AT+MQTTUSERCFG=0,1,\"%s\",\"%s\",\"%s\",0,0,\"\"", BROKER_CLIENT_ID, BROKER_USERNAME,BROKER_USERNAME_PWD);
    ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );    
    
    /* 连接云端 */
    sprintf (cCmd, "AT+MQTTCONN=0,\"%s\",%s,0", BROKER_HOST_IP, BROKER_HOST_PORT);
    ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );    		

    /* 订阅环回消息 */
    sprintf (cCmd, "AT+MQTTSUB=0,\"%s\",0", LOOP_PUB_SUB_TOPIC);
    ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );     
    
    /* 发送环回消息 */
    sprintf (cCmd, "AT+MQTTPUB=0,\"%s\",\"%s\",0,0", LOOP_PUB_SUB_TOPIC, testdata);
    ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );   

    /* 属性上报--订阅消息 */
    sprintf (cCmd, "AT+MQTTSUB=0,\"%s\",0", POST_REPLY_TOPIC);
    ESP8266_Cmd ( cCmd, "OK", NULL, 1000 ); 
    
    /* 属性上报--发布消息 */
    sprintf (cCmd, "AT+MQTTPUB=0,\"%s\",\"%s\",0,0", POST_TOPIC, postdata);
    ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );  

    while (1)
    {
        /* 设置属性--订阅消息 */
        sprintf (cCmd, "AT+MQTTSUB=0,\"%s\",0", SET_PROPERTY_TOPIC);
        ESP8266_Cmd ( cCmd, "OK", NULL, 2000 );          
    }  
}

3.2.1 ESP8266_Cmd函数介绍

        此函数主要实现了向esp8266模块发送AT指令,并且在串口助手打印收到的AT回复,形参reply1和reply2为返回字符串的子集,如果返回字符串存在reply1和reply2这两个参数,那么返回成功。

bool ESP8266_Cmd ( char * cmd, char * reply1, char * reply2, u32 waittime )
{    
	strEsp8266_Fram_Record .InfBit .FramLength = 0;               //从新开始接收新的数据包

	macESP8266_Usart ( "%s\r\n", cmd );

	if ( ( reply1 == 0 ) && ( reply2 == 0 ) )                      //不需要接收数据
		return true;
	
	Delay_ms ( waittime );                 //延时
	
	strEsp8266_Fram_Record .Data_RX_BUF [ strEsp8266_Fram_Record .InfBit .FramLength ]  = '\0';

	macPC_Usart ( "%s", strEsp8266_Fram_Record .Data_RX_BUF );
  
	if ( ( reply1 != 0 ) && ( reply2 != 0 ) )
		return ( ( bool ) strstr ( strEsp8266_Fram_Record .Data_RX_BUF, reply1 ) || 
						 ( bool ) strstr ( strEsp8266_Fram_Record .Data_RX_BUF, reply2 ) ); 
 	
	else if ( reply1 != 0 )
		return ( ( bool ) strstr ( strEsp8266_Fram_Record .Data_RX_BUF, reply1 ) );
	
	else
		return ( ( bool ) strstr ( strEsp8266_Fram_Record .Data_RX_BUF, reply2 ) );
	
}

3.2.2 wifi连接函数介绍

        连接wifi主要通过< AT+CWJAP="wifi帐号","wifi密码"  >指令进行wifi的连接,这里做了简单的处理,既是已经连上wifi下次不必重复连接。

bool ESP8266_JoinAP ( char * pSSID, char * pPassWord )
{
	char cCmd [120];

    sprintf ( cCmd, "AT+CWJAP?");
    /* 如果已经连接wifi,那么不需要重复链接 */
    if (ESP8266_Cmd ( cCmd, "+CWJAP:", NULL, 5000 ) == true)
    {
        return true;
    }
    
    printf("正在连接wifi\r\n");
	sprintf ( cCmd, "AT+CWJAP=\"%s\",\"%s\"", pSSID, pPassWord );
	
	return ESP8266_Cmd ( cCmd, "OK", NULL, 5000 );
	
}

3.2.3 云端连接语句介绍

        云端的连接主要分为两部分: 详情请参见ESP8266 -- 串口助手 AT MQTT 指令与阿里云物联网平台建立通信(三)

一是配置云端参数:配置云端参数主要是< AT+MQTTUSERCFG=0,1,"client_id","username","password",0,0,"" >指令实现。

二是连接云端:通过< AT+MQTTCONN=0,"host",port,0 > 指令实现。

#define BROKER_HOST_IP          "iot-06z12ga2pdtk6xu.mqtt.iothub.aliyuncs.com"      //要连接的服务器的 IP
#define BROKER_HOST_PORT        "1883"                                              //要连接的服务器的端口

/* 对于BROKER_CLIENT_ID定义字符串的解释:
   原本字符串:"izi37b1wleB.TestDevice|securemode=2,signmethod=hmacsha256,timestamp=1701570366200|"
   esp8266支持的字符串:"izi37b1wleB.TestDevice|securemode=2\,signmethod=hmacsha256\,timestamp=1701570366200|"     需在','前加'\'
   C语言基于esp8266支持的字符串:"izi37b1wleB.TestDevice|securemode=2\\,signmethod=hmacsha256\\,timestamp=1701570366200|"  需在'\'前再加一个'\'
   最终通过串口发出去的字符串:"izi37b1wleB.TestDevice|securemode=2\,signmethod=hmacsha256\,timestamp=1701570366200|"
 */
#define BROKER_CLIENT_ID        "izi37b1wleB.TestDevice|securemode=2\\,signmethod=hmacsha256\\,timestamp=1701570366200|"   
#define BROKER_USERNAME         "TestDevice&izi37b1wleB"
#define BROKER_USERNAME_PWD     "77517177642f88688ac2d6286f30699cff7d699daad8394d837608dc5db5935b"

/* 配置云端参数 */
sprintf (cCmd, "AT+MQTTUSERCFG=0,1,\"%s\",\"%s\",\"%s\",0,0,\"\"", BROKER_CLIENT_ID, BROKER_USERNAME,BROKER_USERNAME_PWD);
ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );    

/* 连接云端 */
sprintf (cCmd, "AT+MQTTCONN=0,\"%s\",%s,0", BROKER_HOST_IP, BROKER_HOST_PORT);
ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );    		

3.2.4 环回消息测试语句介绍

        环回消息主要是依赖自定义的发布和接收Topic,首先使用< AT+MQTTSUB=0,"topic",0 >命令订阅消息,再通过< AT+MQTTPUB=0,"topic","data",0,0 >指令回读发送的消息。

const char testdata[] = "This is a MQTT loop test";

#define LOOP_PUB_SUB_TOPIC      "/izi37b1wleB/TestDevice/user/LoopTopic"  //环回测试主题

/* 订阅环回消息 */
sprintf (cCmd, "AT+MQTTSUB=0,\"%s\",0", LOOP_PUB_SUB_TOPIC);
ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );     
    
/* 发送环回消息 */
sprintf (cCmd, "AT+MQTTPUB=0,\"%s\",\"%s\",0,0", LOOP_PUB_SUB_TOPIC, testdata);
ESP8266_Cmd ( cCmd, "OK", NULL, 1000 ); 

3.2.5 属性上报语句介绍

        属性上报和环回消息类似,只是订阅的主题不一样,环回测试主题是自定义的收发Topic,属性上报则是物模型的属性上报Topic。

/* 对于postdata定义字符串的解释:
   原本字符串:                   "{
    (JSON格式)                         "id":1701504913858,
                                       "params":
                                       {
                                           "TestTSL":888
                                       },
                                       "version":"1.0",
                                       "method":"thing.event.property.post"
                                   }"
                                   
   esp8266支持的字符串:          "{
   (也就是在'"'前面增加'\'             \"id\":1701504913858\,
          在','前面增加'\')            \"params\":
                                       {
                                           \"TestTSL\":888
                                       }\,
                                       \"version\":\"1.0\"\,
                                       \"method\":\"thing.event.property.post\"
                                   }"
                              
   C语言基于esp8266支持的字符串: "{
   (如果是','需在'\'前再加1个'\',     \\\"id\\\":1701504913858\\,
    如果是'"'需在'\'前再加2个'\'       \\\"params\\\":
                                       {
                                           \\\"TestTSL\\\":888
                                       }\\,
                                       \\\"version\\\":\\\"1.0\\\"\\,
                                       \\\"method\\\":\\\"thing.event.property.post\\\"
                                   }"
                                   
   最终通过串口发出去的字符串:   "{
                                       \"id\":1701504913858\,
                                       \"params\":
                                       {
                                           \"TestTSL\":888
                                       }\,
                                       \"version\":\"1.0\"\,
                                       \"method\":\"thing.event.property.post\"
                                   }"  
 */
 
const char postdata[] = "{\\\"id\\\":1701504913858\\,\\\"params\\\":{\\\"TestTSL\\\":888}\\,\\\"version\\\":\\\"1.0\\\"\\,\\\"method\\\":\\\"thing.event.property.post\\\"}";

#define POST_TOPIC              "/sys/izi37b1wleB/TestDevice/thing/event/property/post"        //发布
#define POST_REPLY_TOPIC        "/sys/izi37b1wleB/TestDevice/thing/event/property/post_reply"  //订阅

/* 属性上报--订阅消息 */
sprintf (cCmd, "AT+MQTTSUB=0,\"%s\",0", POST_REPLY_TOPIC);
ESP8266_Cmd ( cCmd, "OK", NULL, 1000 ); 

/* 属性上报--发布消息 */
sprintf (cCmd, "AT+MQTTPUB=0,\"%s\",\"%s\",0,0", POST_TOPIC, postdata);
ESP8266_Cmd ( cCmd, "OK", NULL, 1000 );  

3.2.6 设置属性语句介绍

        设置属性就是实时订阅云端的消息,这里2s订阅一次消息。

#define SET_PROPERTY_TOPIC      "/sys/izi37b1wleB/TestDevice/thing/service/property/set"       //订阅

while (1)
{
    /* 设置属性--订阅消息 */
    sprintf (cCmd, "AT+MQTTSUB=0,\"%s\",0", SET_PROPERTY_TOPIC);
    ESP8266_Cmd ( cCmd, "OK", NULL, 2000 );          
}  

4 实验现象

4.1 代码运行现象

ESP8266 -- STM32与阿里云物联网平台建立通信(四)_第1张图片

4.2 属性上报现象  

        这里属性上报的TestTSL=888,我们打开云端,查看到TestTSL的值确实是888。ESP8266 -- STM32与阿里云物联网平台建立通信(四)_第2张图片

 4.3 设置属性现象

        通过阿里云物联网平台,设置TestTSL=123,查看到串口助手显示的值也为123。

ESP8266 -- STM32与阿里云物联网平台建立通信(四)_第3张图片

        注意:此代码不能直接和云端进行通信,需要自己搭建云平台,并修改相对应的云端IP、client_id及密码,云端的搭建可以参考:ESP8266 -- 搭建阿里云物联网与MQTT.fx通信的平台(二)

接下来讲解利用物联网平台读取和控制STM32板载资源,敬请期待。。。

你可能感兴趣的:(esp8266,stm32,阿里云,物联网)