原理 :ESP8266接收到MQTT消息后,利用串口发送给STM32
STM32的代码就是串口控制代码,只不过在这个转口几经周折,从阿里云服务器到MQTT到ESP8266再到STM32的串口

代码分为两个部分,中转站ESP8266,最终处理STM32

ESP8266部分代码:

记得改WIFI 和服务器

#include <ESP8266WiFi.h>
#include <PubSubClient.h>
#define LED D0 // GPIO引脚别名 仅使用于NodeMCU开发板

//默认使用'$'符号作为调试信息输出的标识
//默认使用'#'符号作为控制信息的标识开头

const char *WIFI_SSID = "xxxxx";     // WIFI名称
const char *WIFI_PASS = "xxxx"; // WIFI密码

const char *MQTT_BROKER = "xxxx"; // MQTT服务器地址
const int MQTT_PORT = 1883;             // MQTT服务端口
const char *CLIENT_ID = "c001";         //客户端ID
const char *PUBLISH_TOPIC = "pub01";    //发布的topic
const char *SUBSCRIBE_TOPIC = "sub01";  //订阅的topic

void callback(char *topic, uint8_t *message, unsigned int length); //回调函数声明,用于传入mqtt客户端构造函数作为参数

WiFiClient wifiClient;
//参数: MQTT服务器地址,端口号,回调函数名,承载的连接(WIFI)
PubSubClient mqttClient(MQTT_BROKER, MQTT_PORT, callback, wifiClient);

//回调函数
void callback(char *topic, uint8_t *message, unsigned int length)
{
   // MQTT_doPublishOnDefaultTopic("$Received message:");
    String sTopic = topic; //将topic转换为String 可以加以处理

    String sMessage = (char *)message;        //将消息转换为String
    sMessage = sMessage.substring(0, length); //取合法长度 避免提取到旧消息
    //MQTT发布函数
    //MQTT_doPublishOnDefaultTopic(sMessage);
    
    //按字符判断mqtt消息中命令的作用 可以自行定义
    何顺                                         🔥 2020/8/23 22:23:14
 47.93.117.91:18083

何顺                                         🔥 2020/8/24 9:06:53
阿里云「在家实践」活动,三月初就已经有了,ECS云服务器领取地址:https://developer.aliyun.com/adc/student/

一次可领取6个月,快到期的时候可以再免费续期一次6个月,加起来可以免费用一年(12个月)。

下面是测试题和答案:

【腾讯文档】阿里云「在家实践」测试题及答案
https://docs.qq.com/doc/DVktVWFpUQ0FxQUR5

何顺                                         🔥 2020/8/24 9:57:16
阿里云「在家实践」活动,三月初就已经有了,ECS云服务器领取地址:https://developer.aliyun.com/adc/student/

一次可领取6个月,快到期的时候可以再免费续期一次6个月,加起来可以免费用一年(12个月)。

下面是测试题和答案:

【腾讯文档】阿里云「在家实践」测试题及答案
https://docs.qq.com/doc/DVktVWFpUQ0FxQUR5

何顺                                         🔥 2020/8/24 9:57:53


隔壁老黄 2020/8/26 19:28:05


隔壁老黄 2020/8/26 19:28:11
搞个給

何顺                                         🔥 2020/8/26 21:36:07
void TIM3_PWM_Init(u16 arr,u16 psc)
{  
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef  TIM_OCInitStructure;
	

	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);	//使能定时器3时钟
 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB  | RCC_APB2Periph_AFIO, ENABLE);  //使能GPIO外设和AFIO复用功能模块时钟
	
	GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE); //Timer3部分重映射  TIM3_CH2->PB5    
 
   //设置该引脚为复用输出功能,输出TIM3 CH2的PWM脉冲波形	GPIOB.5
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5; //TIM_CH2
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;  //复用推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);//初始化GPIO
 
   //初始化TIM3
	TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
	TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值 
	TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
	
	//初始化TIM3 Channel2 PWM模式	 
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2; //选择定时器模式:TIM脉冲宽度调制模式2
 	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; //输出极性:TIM输出比较极性高
	TIM_OC2Init(TIM3, &TIM_OCInitStructure);  //根据T指定的参数初始化外设TIM3 OC2

	TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable);  //使能TIM3在CCR2上的预装载寄存器
 
	TIM_Cmd(TIM3, ENABLE);  //使能TIM3

何顺                                         🔥 2020/8/26 23:42:43


何顺                                         🔥 2020/8/27 1:25:29
还没上来啊

何顺                                         🔥 2020/8/27 1:25:45


何顺                                         🔥 2020/8/27 1:26:15
别带了别带了

何顺                                         🔥 2020/8/27 9:16:46


何顺                                         🔥 2020/8/28 20:41:07


何顺                                         撤回了一条消息

何顺                                         🔥 2020/8/29 10:47:59


何顺                                         🔥 2020/8/29 10:49:12


何顺                                         🔥 2020/8/30 22:13:13


何顺                                         🔥 2020/8/31 15:45:30


何顺                                         🔥 2020/8/31 16:30:56


何顺                                         🔥 15:04:16
之前我们通过MQTTX控制ESP8266开关灯
[ESP8266阿里云开关灯](https://blog.csdn.net/qq_40985093/article/details/108206804)
这一节我们用MQTT控制STM32的小灯。

<font color = red size=6  >    原理 :ESP8266接收到MQTT消息后,利用串口发送给STM32</font>
STM32的代码就是串口控制代码,只不过在这个转口几经周折,从阿里云服务器到MQTT到ESP8266再到STM32的串口

代码分为两个部分,中转站ESP8266,最终处理STM32

## ESP8266部分代码:

<font color = red size=6  >  记得改WIFI  和服务器</font>
```cpp
#include <ESP8266WiFi.h>
#include <PubSubClient.h>
#define LED D0 // GPIO引脚别名 仅使用于NodeMCU开发板

//默认使用'$'符号作为调试信息输出的标识
//默认使用'#'符号作为控制信息的标识开头

const char *WIFI_SSID = "xxxxx";     // WIFI名称
const char *WIFI_PASS = "xxxx"; // WIFI密码

const char *MQTT_BROKER = "xxxx"; // MQTT服务器地址
const int MQTT_PORT = 1883;             // MQTT服务端口
const char *CLIENT_ID = "c001";         //客户端ID
const char *PUBLISH_TOPIC = "pub01";    //发布的topic
const char *SUBSCRIBE_TOPIC = "sub01";  //订阅的topic

void callback(char *topic, uint8_t *message, unsigned int length); //回调函数声明,用于传入mqtt客户端构造函数作为参数

WiFiClient wifiClient;
//参数: MQTT服务器地址,端口号,回调函数名,承载的连接(WIFI)
PubSubClient mqttClient(MQTT_BROKER, MQTT_PORT, callback, wifiClient);

//回调函数
void callback(char *topic, uint8_t *message, unsigned int length)
{
   // MQTT_doPublishOnDefaultTopic("$Received message:");
    String sTopic = topic; //将topic转换为String 可以加以处理

    String sMessage = (char *)message;        //将消息转换为String
    sMessage = sMessage.substring(0, length); //取合法长度 避免提取到旧消息
    //MQTT发布函数
    //MQTT_doPublishOnDefaultTopic(sMessage);
    
    //按字符判断mqtt消息中命令的作用 可以自行定义

何顺                                         🔥 15:04:18
    if (sMessage.charAt(0) == '#')
    {         //第一位#
        if (sMessage.charAt(1) == 'D') 
        {     //第一位l
            if (sMessage.charAt(2) == '1') 
            { //第一位1
                //处理#D1
                digitalWrite(LED, 0);  //输出低电平
                  digitalWrite(LED_BUILTIN, LOW);  
                    MQTT_doPublishOnDefaultTopic("open\r\n");
                    Serial.println("#D1\r\n");
            } 
            else if (sMessage.charAt(2) == '0') 
            {
                //处理#D0
                digitalWrite(LED, 1); //初始化输出高电平
                 digitalWrite(LED_BUILTIN, HIGH);  
                  MQTT_doPublishOnDefaultTopic("close\r\n");
                  Serial.println("#D0\r\n");
            }
        }
        if (sMessage.charAt(1) == 'd') 
        {
            //处理#d...
        }
    }
}

// MQTT发布函数 需要使用可以解除注释 传入一个字符串 发送到默认配置的发布topic
void MQTT_doPublishOnDefaultTopic(String payload) {
    //参数: 发布的话题,发布的内容
    mqttClient.publish(PUBLISH_TOPIC, payload.c_str()); // String的c_str()方法将一个字符串转换为字符数组
}

//所有Arduino程序都具备的配置函数 系统初始化后只执行一次
void setup() {
    pinMode(LED, OUTPUT); //初始化引脚
    pinMode(LED_BUILTIN, OUTPUT);
    digitalWrite(LED, 1); //初始化输出高电平
     digitalWrite(LED_BUILTIN, HIGH);  

    //启用串口
    Serial.begin(115200);
    // Serial.println("$Hello world!"); //串口输出消息

    //连接WIFI
    WiFi.begin(WIFI_SSID, WIFI_PASS);

    //连接未成功时 循环等待
    while (WiFi.status() != WL_CONNECTED)
    {
        delay(1000);
        Serial.println("$Still waiting...");
    }

    Serial.println("$Wi-Fi connected");
    Serial.print("$IP address: ");
    Serial.println(WiFi.localIP());

    //进行MQTT连接
    //当MQTT服务器连接不成功时
    while (!mqttClient.connected())
    {
        Serial.println("$Waiting for MQTT");
        //执行(重试)MQTT连接
        if (mqttClient.connect(CLIENT_ID)) 
        {
            //连接成功后 订阅指定的topic
            if (mqttClient.subscribe(SUBSCRIBE_TOPIC, 0))
            { // QoS: 0/1/2 [|0: 只发送一次(常用)|1: 至少发送一次|2: 一定收到一次]
                //Serial.println("$Subscribed."); //订阅成功 发送调试消息
                MQTT_doPublishOnDefaultTopic("$Subscribed.");
            } 
            else
            {
                Serial.println("$Subscribe failed."); //订阅不成功 发送调试消息
                //如有需要 可以加死循环阻止程序继续运行
            }
        }
        delay(1000);
    } 
}

//所有Arduino程序都具备的循环函数 setup()函数执行后循环执行
void loop()
{
    delay(500);
    mqttClient.loop(); //保证MQTT客户端持续运行和接收消息
}

STM32部分代码

直接用正点原子的串口使用代码,main函数改成如下部分即可。

#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"
 
/************************************************
 ALIENTEK精英STM32开发板实验4
 串口 实验   
 技术支持:www.openedv.com
 作者:正点原子 @ALIENTEK
************************************************/
 int main(void)
 {		
	delay_init();	    	 //延时函数初始化	  
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置NVIC中断分组2:2位抢占优先级,2位响应优先级
	uart_init(115200);	 //串口初始化为115200
 	LED_Init();			     //LED端口初始化
	KEY_Init();          //初始化与按键连接的硬件接口
 	while(1)
	{
		if(USART_RX_STA&0x8000)
		{					   
					if(USART_RX_BUF[0]=='#')
				{
					
					if(USART_RX_BUF[1]=='D')
					{
						if(USART_RX_BUF[2]=='1')
						{
							LED1=1;
							LED0=0;
						}
							if(USART_RX_BUF[2]=='0')
						{
							LED1=0;
							LED0=1;
						}
						
					}
				}
			USART_RX_STA=0;
		}

	}	 
 }

esp8266 thonny开关使用_#include