首页 > 其他分享 >STM32+ESP8266-连接阿里云-创建云产品流转实现STM32与Android app通讯(1)

STM32+ESP8266-连接阿里云-创建云产品流转实现STM32与Android app通讯(1)

时间:2024-07-27 11:58:53浏览次数:26  
标签:char esp8266 app STM32 printf ESP8266 log

前言

本文章的内容为STM32通过ESP8266利用AT指令连接阿里云平台,并创建设备和创建云产品流转主题,来为实现Android app与STM32的发送接收数据做准备。Android app的实现由于篇幅不宜过长,将放到下一篇文章中。

演示视频

实现一个简单的app来控制stm32开关灯、蜂鸣器、门(舵机),显示温湿度(DTH11模块)数据,光度数据。

话不多说先看实验效果:

<iframe allowfullscreen="true" data-mediaembed="bilibili" frameborder="0" id="bDPHFi0i-1722033920375" src="https://player.bilibili.com/player.html?aid=112852178309078"></iframe>

基于STM32+esp8266+freertos+Android app+阿里云的智能家居系统


一、硬件设计:

接线方式

本项目所使用到的WIFI模块为ESP8266-01S,接线方式如下:注意TXD和RXD的接线方式,很多时候接了模块却没有反应,就是因为硬件连线没有接对。

STM32/USBTTLesp8266
GND         GND
3.3V3.3V
TXDRXD
RXDTXD


二、软件设计 

 烧录MQTT固件 

需要有一个串口TTL模块,硬件连接如下,因为我在烧录固件的时候,esp8266总是显示等待上电同步,无法正常烧写固件,通过查阅资料和尝试之后,通过将ESP8266的3V3和EN与串口TTL模块的3V3连接,GND和IO0引脚共同连接到GND上,不懂怎么做的小伙伴,可以看看我下面的图片去将杜邦线进行修改即可。

TTL模块 ESP8266
3V3        3V3 和 EN
GNDGND 和 IO0
TXDRXD
RXDTXD

ESP8266-01S的MQTT固件和烧写固件的软件在下方链接中,请自行下载:

链接:https://pan.baidu.com/s/1f5cNlNCMdW22GDBMksK59w?pwd=76kn 
提取码:76kn 
--来自百度网盘超级会员V5的分享

打开ESPFlashDownloadTool之后,选择第一个框框ESP8266 DownloadTool

第一步先选择MQTT固件,烧写到ESP8266的0x00000地址。第二步先ERASE,第三步再点击Start进行固件烧写即可。如果遇到程序下载不进去的情况,可能需要将RST在下载的过程中接地1~2秒然后浮空。


阿里云创建设备

在阿里云主页搜索框搜索物联网平台,或者直接点击此链接进入:物联网平台 (aliyun.com) 

如果没有开通公共实例的需要先开通,点击进入。

点击左栏的设备管理中产品选项的新建产品:1.自定义产品名称,2.勾选标准品类,3.节点类型选择直连设备,4.联网方式:WI-FI,5.数据格式:ICA 标准数据格式,6.点击确认即可。

创建后点击操作中的管理设备,进行设备添加。

输入设备名称和备注名称(方便区分设备)

如果用于App和STM32通讯,需要为其各创建一个设备,根据上面的设备创建流程走即可。 


创建云流转主题

创建自定义主题:在同一个产品下新建两个自定义的Topic:一个用于STM32上传传感器数据到App,另一个用于App下发命令控制STM32。创建的流程如下,设备操作权限记得选择发布和订阅。

这是我创建的两个主题:一个是STM32_to_Android 用于STM32上传传感器数据到App,另一个是Android_to_STM32用于Android app下发命令控制STM32。后面会看到,因为主题选择了发布和订阅模式,所以我通过将STM32和App都上传到同一个主题,也能够转发到两个设备。

 创建云产品流转:下面创建了两个规则:

转发数据配置:通过以下配置,我们就可以从数据流传到的topic中接收到设备上传的数据了,可以选择自定义topic也可以选择物模型数据下发的指定topic。

创建完后,记得将每个规则启动。


ESP8266模块驱动

esp8266_mqtt.c文件中,我们需要将ESP8266_Init()在main函数之前进行调用,通过一系列AT指令来对ESP8266进行初始化,使其能够连接WIFI和阿里云服务器。 

连接参数在设备中的MQTT连接参数,点击查看就能看到。订阅的主题是云产品流转中用于转发数据的主题。我们通过以下AT指令来让STM32连接到阿里云,并订阅Android App用于控制STM32发布的主题。参数的使用在以下AT指令中都有详细的注释。

//复位
AT+RST

//配置wifi模式 STA模式
AT+CWMODE=1

//wifi 名和密码
AT+CWJAP="360","02744871"   

//username、passwd
AT+MQTTUSERCFG=0,1,"NULL","STM32&k1h2hJkoTA7","ff9794cb37d57d38fb4a7e2b9091c7e1ade031fd47c3868efdb571e448b3a65a",0,0,""

//clientid
AT+MQTTCLIENTID=0,"k1h2hJkoTA7.STM32|securemode=2\,signmethod=hmacsha256\,timestamp=1721776877786|"

//mqttHostUrl:iot-06z00jlxn39wea0.mqtt.iothub.aliyuncs.com
AT+MQTTCONN=0,"iot-06z00jlxn39wea0.mqtt.iothub.aliyuncs.com",1883,1

//订阅主题
AT+MQTTSUB=0,"/sys/k1h2hJkoTA7/STM32/thing/service/property/set\",1

esp8266_mqtt.c文件中的函数功能如下:

  ESP8266_Init()用于发送AT指令连接阿里云

  Send_Data_To_Cloud() 发送传感器数据到k1h2hJkoTA7/STM32/user/Android_STM32主题,我们在先前已经在云产品流转中将其转发到/k1h2hJkoTA7/Android_app/user/STM32toAndroid主题,App将订阅此主题获取传感器数据显示在屏幕上。

  Get_Data_From_Cloud()用于获取先前ESP8266_Init()订阅的/sys/k1h2hJkoTA7/STM32/thing/service/property/set主题所发布的信息,用于接收APP下发的指令。

#include "esp8266_mqtt.h"

#define ESP8266_WIFI_INFO		"AT+CWJAP=\"360\",\"02744871\"\r\n"   //连接的Wifi名 密码
#define MQTTUSERCFG  "AT+MQTTUSERCFG=0,1,\"NULL\",\"STM32&k1h2hJkoTA7\",\"25940a10a11126eb67377bed21b8983b595b5c96429664f9519baeaa5f91dd58\",0,0,\"\"\r\n"
#define MQTTCLIENTID "AT+MQTTCLIENTID=0,\"k1h2hJkoTA7.STM32|securemode=2\\,signmethod=hmacsha256\\,timestamp=1721780638696|\"\r\n"
#define MQTTCONN "AT+MQTTCONN=0,\"iot-06z00jlxn39wea0.mqtt.iothub.aliyuncs.com\",1883,1\r\n"



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

extern   UART_HandleTypeDef huart3;
#define  ESP8266_USART      &huart3
/*
************************************************************
*	函数名称:	Usart_SendString
*
*	函数功能:	串口数据发送
*
*	入口参数:	USARTx:串口组
*				str:要发送的数据
*				len:数据长度
*
*	返回参数:	无
*
*	说明:		
************************************************************
*/
void Usart_SendString(UART_HandleTypeDef *USARTx, unsigned char *str, unsigned short len)
{
	unsigned short count = 0;
	
	if (str == NULL || USARTx == NULL) 
	{
			return;
	}
	for (; count < len; count++) 
	{
		if (str[count] == '\0') 
		{
				break;
		}
		HAL_UART_Transmit(USARTx, (uint8_t *)(str + count), 1, 10);
	}
}

//==========================================================
//	函数名称:	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;
	
	printf("\r\nWifiTx = %s\n", cmd);
	HAL_UART_Transmit(&huart3, (uint8_t *)cmd, strlen((const char *)cmd), 0xffff);
	
	while(timeOut--)
	{
		if(ESP8266_WaitRecive() == REV_OK)							//如果收到数据
		{
			//printf("esp8266_buf : %s\n", esp8266_buf);
			if(strstr((const char *)esp8266_buf, res) != NULL)		//如果检索到关键词
			{
				ESP8266_Clear();									//清空缓存
				
				return 0;
			}
		}
		HAL_Delay(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(ESP8266_USART, data, len);		//发送设备连接请求数据
		HAL_UART_Transmit(&huart3, (uint8_t *)data, len, 0xffff);
	}

}

//==========================================================
//	函数名称:	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)								//如果接收完成
		{
			printf("%s\n", esp8266_buf);
			ptrIPD = strstr((char *)esp8266_buf, "+MQTTSUBRECV:");				//搜索“IPD”头
			if(ptrIPD != NULL)											//如果没找到,可能是IPD头的延迟,还是需要等待一会,但不会超过设定的时间
			{
				ptrIPD = strchr(ptrIPD, '{');							//找到':'
				if(ptrIPD != NULL)
				{
					return (unsigned char *)(ptrIPD);
				}
			}
		}
		HAL_Delay(1);													//延时等待
	} while(timeOut--);
	ESP8266_Clear();
	return NULL;														//超时还未找到,返回空指针
}

//==========================================================
//	函数名称:	ESP8266_Init
//
//	函数功能:	初始化ESP8266
//
//	入口参数:	无
//
//	返回参数:	无
//
//	说明:		
//==========================================================
void ESP8266_Init(void)
{
	uint8_t log_line = 32;
	
	HAL_Delay(500);
	ESP8266_Clear();
	ESP8266_SendCmd("AT+RST\r\n", "OK");
	
	printf("1. AT\r\n");
	while(ESP8266_SendCmd("AT\r\n", "OK")) 
	{
		HAL_Delay(500);
		printf("Sending AT command failed, retrying...\r\n");		
	}
	LCD_print_log(0, log_line*0, (uint8_t *)"1.AT");
	
	printf("2. CWMODE\r\n");
	while(ESP8266_SendCmd("AT+CWMODE=1\r\n", "OK"))  //配置为STA模式
	{
		HAL_Delay(500);
		printf("Setting CWMODE failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*1, (uint8_t *)"2.AT+CWMODE");
	
	printf("3. AT+CIPSNTPCFG\r\n");
	while(ESP8266_SendCmd("AT+CIPSNTPCFG=1,8,\"cn.ntp.org.cn\",\"ntp.sjtu.edu.cn\"\r\n", "OK")) 
	{
		HAL_Delay(500);
		printf("Setting CWDHCP failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*2, (uint8_t *)"3.AT+CIPSNTPCFG");
	
	printf("4. CWJAP\r\n");
	while(ESP8266_SendCmd(ESP8266_WIFI_INFO, "GOT IP")) 
	{
		HAL_Delay(500);
		printf("Connecting to WiFi failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*3, (uint8_t *)"4.AT+CWJAP");
	
	//设备名、密码
	printf("5.AT+MQTTUSERCFG\r\n");
	while(ESP8266_SendCmd(MQTTUSERCFG, "OK")) 
	{
		HAL_Delay(500);
		//printf("Starting TCP connection failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*4, (uint8_t *)"5.AT+MQTTUSERCFG");
	
	printf("6.AT+MQTTCLIENTID\r\n");
	while(ESP8266_SendCmd(MQTTCLIENTID, "OK")) 
	{
		HAL_Delay(500);
		//printf("Starting TCP connection failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*5, (uint8_t *)"6.AT+MQTTCLIENTID");
	
	printf("7.AT+MQTTCONN\r\n");
	while(ESP8266_SendCmd(MQTTCONN, "OK")) 
	{
		HAL_Delay(500);
		printf("MQTTCONN failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*6, (uint8_t *)"7.AT+MQTTCONN");
	
	LCD_Clear(WHITE); //清屏
	printf("8.AT+MQTTSUB\r\n");
	            while(ESP8266_SendCmd("AT+MQTTSUB=0,\"/sys/k1h2hJkoTA7/STM32/thing/service/property/set\",1\r\n", "OK")) 
	{
		HAL_Delay(500);
		printf("MQTTSUB failed, retrying...\r\n");
	}
	LCD_print_log(0, log_line*0, (uint8_t *)"8.AT+MQTTSUB");
	LCD_print_log(0, log_line*1, (uint8_t *)"Cloud connected");
	
	printf("Cloud connection successful\n");
}

void Get_Data_From_Cloud(void) 
{
    unsigned char *data = ESP8266_GetIPD(200);
    if (data != NULL) 
    {
        // 处理接收到的数据,例如:
        printf("Received data: %s\n", data);

        // 根据需要解析 JSON 数据
        cJSON *json = cJSON_Parse((char *)data);
        if (json != NULL) 
        {
            // 解析并处理 LED 数据
            cJSON *led = cJSON_GetObjectItem(json, "LED");
            if (cJSON_IsNumber(led)) 
            {
                printf("LED: %d\n", led->valueint);
								LED_TOGGLE(1);
                //在这里处理 LED 数据
            } 
            else 
            {
                printf("Error: LED is not a number\n");
            }

            // 解析并处理 door 数据
            cJSON *door = cJSON_GetObjectItem(json, "door");
            if (cJSON_IsNumber(door)) 
            {
                printf("door: %d\n", door->valueint);
								Toggle_Door();
                // 在这里处理 door 数据
            } 
            else 
            {
                printf("Error: door is not a number\n");
            }

            // 解析并处理 beep 数据
            cJSON *beep = cJSON_GetObjectItem(json, "beep");
            if (cJSON_IsNumber(beep)) 
            {
              printf("beep: %d\n", beep->valueint);
							BEEP_TOGGLE(1,100);
                // 在这里处理 beep 数据
            } 
            else 
            {
                printf("Error: beep is not a number\n");
            }

            cJSON_Delete(json);
        } 
        else 
        {
            printf("Error: Failed to parse JSON\n");
        }
    } 
    else 
    {
        //printf("Error: No data received\n");
    }
}


extern float Light_Value;
extern unsigned int rec_data[4];


void Send_Data_To_Cloud(void)
{
	char buf[256];
	int i = 0;
	
	if(rec_data[2] > 0 && rec_data[2] <=200)
	{
		//自定义云产品转发上传主题
		sprintf(buf, "AT+MQTTPUB=0,\"/k1h2hJkoTA7/STM32/user/Android_STM32\",\"{\\\"params\\\":{\\\"Temp\\\":%d}}\",1,0\r\n", rec_data[2]);
		printf("WifiTx = %s", buf);
		HAL_UART_Transmit(&huart3, (uint8_t *)buf, strlen((const char *)buf), 0xffff);
		memset(buf, 0, sizeof(buf));
		ESP8266_Clear();									//清空缓存
		HAL_Delay(100);
	}

	if(rec_data[0] > 0 && rec_data[0] <=200)
	{
		sprintf(buf, "AT+MQTTPUB=0,\"/k1h2hJkoTA7/STM32/user/Android_STM32\",\"{\\\"params\\\":{\\\"Humi\\\":%d}}\",1,0\r\n", rec_data[0]);
		printf("WifiTx = %s", buf);
		HAL_UART_Transmit(&huart3, (uint8_t *)buf, strlen((const char *)buf), 0xffff);
		memset(buf, 0, sizeof(buf));
		ESP8266_Clear();									//清空缓存
		HAL_Delay(100);
	}
	
	if(Light_Value > 0 && Light_Value <=200)
	{
		sprintf(buf, "AT+MQTTPUB=0,\"/k1h2hJkoTA7/STM32/user/Android_STM32\",\"{\\\"params\\\":{\\\"light\\\":%d}}\",1,0\r\n", (uint8_t)Light_Value);
		printf("WifiTx = %s", buf);
		HAL_UART_Transmit(&huart3, (uint8_t *)buf, strlen((const char *)buf), 0xffff);
		memset(buf, 0, sizeof(buf));
		ESP8266_Clear();									//清空缓存
		HAL_Delay(100);
	}
}

结尾 

完整代码可在gzh搜索嵌入式crafter,发送关键词:STM32_ESP8266_阿里云获得网盘链接。

标签:char,esp8266,app,STM32,printf,ESP8266,log
From: https://blog.csdn.net/MOS_JBET/article/details/140729208

相关文章

  • Hbuilder发布app应用流程
    使用HBuilder发布App应用的流程可以分为以下几个步骤。HBuilder是一款高效的前端开发工具,特别适合移动端应用的开发。以下是详细的步骤:1.开发准备安装HBuilder:从DCloud官方网站下载并安装HBuilder。创建项目:打开HBuilder,新建一个移动App项目,选择合适的模板进行开......
  • STM32开发环境配置记录——关于PlatformIO + VSCode + CubeMX的集成环境配置
    前言​ 为什么配置这样的一个环境呢?鄙人受够了Keil5那个简陋的工作环境了,实在是用不下去,调试上很容易跟CubeMX的代码产生不协调导致调试——发布代码不一致造成的一系列问题。CubeIDE虽说不错,但是它的代码辅助功能和构建系统实在不敢恭维,经常出现Makefile未同步导致符号定义冲突,......
  • 质押托管模式APP开发/智能合约DAPP系统开发搭建
    质押托管模式APP的开发是一个涉及多个技术环节和业务流程的复杂过程。以下是一个概括性的开发指南,旨在帮助理解质押托管模式APP的基本框架和开发要点:一、需求分析明确业务场景:首先,需要明确质押托管模式APP的业务场景,包括质押资产的种类(如加密货币、传统金融资产等)、质押流程......
  • 速通——条件构造器(Wrapper)
    在MyBatis-Plus中,Wrapper类是构建查询和更新条件的核心工具。利用MyBatis-Plus的Wrapper用于构建复杂的数据库查询条件。允许链式调用。核心条件构造器Wrapper的层次结构为Wrapper:条件构造抽象类,最顶端父类-AbstractWrapper:用于查询条件封装,生成sql的where条件......
  • 零基础STM32单片机编程入门(二十二) ESP8266 WIFI模块实战含源码
    文章目录一.概要二.ESP8266WIFI模块主要性能参数三.ESP8266WIFI模块芯片内部框图四.ESP8266WIFI模块原理图五.ESP8266WIFI模块与单片机通讯方法1.硬件连接2.ESP8266模块AT指令介绍六.STM32单片机与ESP8266WIFI模块通讯实验1.硬件准备2.软件工程3.软件主要代码4.实验......
  • uniapp(安卓苹果app端) - 微信小程序分享提示报错“由于不支持的分享类型无法分享到微
    问题说明在uni-app(手机app端)安卓Android、苹果ios系统中,解决分享微信小程序时,页面提示:由于不支持的分享类型无法分享到微信|由于应用和小程序未绑定在同意微信开放平台账号,无法分享到微信,uniApp开发App分享微信时出现无法分享且报错提示、分享不可用、微信appid、配置......
  • uniapp 手写签名上传服务器
    用的框架是yinghuo,上传用了封装的上传<template><viewclass="container"><jp-signatureref="signatureRef":openSmooth="true"></jp-signature><viewclass="dis-flexm-top20"&......
  • [二、状态管理]3管理应用拥有的状态(3)AppStorage:应用全局的UI状态存储
    AppStorage是应用全局的UI状态存储,是和应用的进程绑定的,由UI框架在应用程序启动时创建,为应用程序UI状态属性提供中央存储。和AppStorage不同的是,LocalStorage是页面级的,通常应用于页面内的数据共享。而AppStorage是应用级的全局状态共享,还相当于整个应用的“中枢”,持久化数据Pers......
  • MapperStruct 嵌套模型中 List<> 转 List<String>
    废话不多说,上代码 宗旨:将List<A>映射为List<String>一,实体类Source//Source中有一个List<A>publicclassSource{privateStringid;privateStringfrom;privateList<A>to;}//A对象中有个BpublicclassA{privateBb;}//B有个addre......
  • Appium各种定位及操作
    前提设置:fromappiumimportwebdriver#模拟器/真机已经被电脑识别到(adbdevices)#字典#1、设置终端参数项desired_caps={"platformName":"Android","platformVersion":"6.1.1","appPackage":"com.ss.android.ar......