STM32与ESP8266的使用
串口透传
- “透传”通常指的是数据的透明传输,意思是在不对数据进行任何处理或修改的情况下,将数据从一个接口转发到另一个接口。
- 值得注意的是要避免串口之间无限制的透明,可以采用互斥锁的方式进行限制
- 使用方法
- 对USART1和USART3(用他俩举例)的模式都是设置为Asynchronous,并开启对应的中断。
- RCC的High SPeed CLock模式设置为Crystal/Ceramic
- 配置对应的时钟为64Mhz
- 在main函数中启动串口1和串口3的空闲中断模式,接收数据 HAL_UARTEx_ReceiveToIdle_IT(&huart1, rxbuf1, sizeof(rxbuf1));
HAL_UARTEx_ReceiveToIdle_IT(&huart3, rxbuf3, sizeof(rxbuf3)); - 在
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef* huart, uint16_t Size)写对应的透传即可
- 代码示例
#include "main.h"
#include <string.h>UART_HandleTypeDef huart1; // 定义串口1的句柄
UART_HandleTypeDef huart3; // 定义串口3的句柄char rxbuf1[128] = {0}; // 用于接收串口1数据的缓冲区
char rxbuf3[128] = {0}; // 用于接收串口3数据的缓冲区
uint8_t uart1_to_uart3_enable = 1; // 控制串口1是否允许发送数据到串口3的标志位
uint8_t uart3_to_uart1_enable = 1; // 控制串口3是否允许发送数据到串口1的标志位/*** @brief 串口接收中断回调函数* 该函数在接收到数据后触发,并根据当前接收的是串口1还是串口3的数据,进行对应的处理。* @param huart 串口句柄* @param Size 接收到的数据大小*/
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef* huart, uint16_t Size)
{printf("HAL_UARTEx_RxEventCallback triggered\n");// 判断当前接收的数据是否来自串口1,并且标志位允许发送给串口3if (huart == &huart1 && uart1_to_uart3_enable){uart3_to_uart1_enable = 0; // 禁止串口3将数据回传给串口1,防止死循环HAL_UART_Transmit(&huart3, (uint8_t*)rxbuf1, Size, HAL_MAX_DELAY); // 将串口1接收的数据发送给串口3memset(rxbuf1, 0, sizeof(rxbuf1)); // 清空串口1接收缓冲区HAL_UARTEx_ReceiveToIdle_IT(&huart1, rxbuf1, sizeof(rxbuf1)); // 重新启动串口1接收中断uart3_to_uart1_enable = 1; // 允许串口3传输数据到串口1printf("port1 sent to port 3\n");}// 判断当前接收的数据是否来自串口3,并且标志位允许发送给串口1else if (huart == &huart3 && uart3_to_uart1_enable){uart1_to_uart3_enable = 0; // 禁止串口1将数据回传给串口3,防止死循环HAL_UART_Transmit(&huart1, (uint8_t*)rxbuf3, Size, HAL_MAX_DELAY); // 将串口3接收的数据发送给串口1memset(rxbuf3, 0, sizeof(rxbuf3)); // 清空串口3接收缓冲区HAL_UARTEx_ReceiveToIdle_IT(&huart3, rxbuf3, sizeof(rxbuf3)); // 重新启动串口3接收中断uart1_to_uart3_enable = 1; // 允许串口1传输数据到串口3printf("port3 sent to port 1\n");}return;
}/*** @brief 重定向printf函数,将其输出重定向到串口1* @param ch 需要输出的字符* @return 返回输出的字符*/
int __io_putchar(int ch)
{HAL_UART_Transmit(&huart1, (unsigned char*)&ch, 1, HAL_MAX_DELAY); // 将字符通过串口1发送return ch; // 返回字符
}/*** @brief 主函数*/
int main(void)
{HAL_Init(); // 初始化HAL库SystemClock_Config(); // 配置系统时钟MX_GPIO_Init(); // 初始化GPIOMX_USART1_UART_Init(); // 初始化串口1MX_USART3_UART_Init(); // 初始化串口3// 启动串口1和串口3的空闲中断模式,接收数据HAL_UARTEx_ReceiveToIdle_IT(&huart1, rxbuf1, sizeof(rxbuf1));HAL_UARTEx_ReceiveToIdle_IT(&huart3, rxbuf3, sizeof(rxbuf3));while (1){// 主循环中可以进行其他任务处理}
}/*** @brief 配置系统时钟* 设置MCU的时钟源、倍频系数等,保证系统运行在正确的时钟频率*/
void SystemClock_Config(void)
{RCC_OscInitTypeDef RCC_OscInitStruct = {0}; // 配置RCC振荡器参数RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; // 配置RCC时钟源及分频系数RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; // 使用外部高速振荡器(HSE)RCC_OscInitStruct.HSEState = RCC_HSE_ON; // 开启HSERCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1; // HSE预分频值为1RCC_OscInitStruct.HSIState = RCC_HSI_ON; // 开启内部高速振荡器(HSI)RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; // 开启PLLRCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; // PLL时钟源为HSERCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL8; // PLL倍频系数为8if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){Error_Handler(); // 时钟配置失败,进入错误处理函数}RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; // 配置不同的时钟类型RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; // 系统时钟源设置为PLLRCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; // AHB时钟不分频RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; // APB1时钟分频系数为2RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; // APB2时钟不分频if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK){Error_Handler(); // 时钟配置失败,进入错误处理函数}
}/*** @brief 初始化串口1* 配置波特率、数据位、停止位等串口参数*/
static void MX_USART1_UART_Init(void)
{huart1.Instance = USART1; // 设置串口1的实例huart1.Init.BaudRate = 115200; // 波特率设置为115200huart1.Init.WordLength = UART_WORDLENGTH_8B; // 数据位长度为8位huart1.Init.StopBits = UART_STOPBITS_1; // 停止位为1位huart1.Init.Parity = UART_PARITY_NONE; // 无校验位huart1.Init.Mode = UART_MODE_TX_RX; // 使能接收和发送模式huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; // 不使用硬件流控huart1.Init.OverSampling = UART_OVERSAMPLING_16; // 过采样设置为16倍if (HAL_UART_Init(&huart1) != HAL_OK){Error_Handler(); // 初始化失败,进入错误处理函数}
}/*** @brief 初始化串口3* 配置波特率、数据位、停止位等串口参数*/
static void MX_USART3_UART_Init(void)
{huart3.Instance = USART3; // 设置串口3的实例huart3.Init.BaudRate = 115200; // 波特率设置为115200huart3.Init.WordLength = UART_WORDLENGTH_8B; // 数据位长度为8位huart3.Init.StopBits = UART_STOPBITS_1; // 停止位为1位huart3.Init.Parity = UART_PARITY_NONE; // 无校验位huart3.Init.Mode = UART_MODE_TX_RX; // 使能接收和发送模式huart3.Init.HwFlowCtl = UART_HWCONTROL_NONE; // 不使用硬件流控huart3.Init.OverSampling = UART_OVERSAMPLING_16; // 过采样设置为16倍if (HAL_UART_Init(&huart3) != HAL_OK){Error_Handler(); // 初始化失败,进入错误处理函数}
}/*** @brief GPIO初始化* 配置GPIO端口,用于其他硬件外设,如LED、按键等*/
static void MX_GPIO_Init(void)
{GPIO_InitTypeDef GPIO_InitStruct = {0};// 启用GPIO时钟__HAL_RCC_GPIOD_CLK_ENABLE();__HAL_RCC_GPIOB_CLK_ENABLE();__HAL_RCC_GPIOC_CLK_ENABLE();__HAL_RCC_GPIOA_CLK_ENABLE();// 配置PC6、PC7、PC8引脚为推挽输出模式,初始化为低电平,用于驱动LED或其他外设HAL_GPIO_WritePin(GPIOC, GPIO_PIN_6 | GPIO_PIN_7 | GPIO_PIN_8, GPIO_PIN_RESET);GPIO_InitStruct.Pin = GPIO_PIN_6 | GPIO_PIN_7 | GPIO_PIN_8; // 设置要配置的引脚GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // 配置为推挽输出模式GPIO_InitStruct.Pull = GPIO_NOPULL; // 不使用上拉或下拉电阻GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; // 设置为低速HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); // 初始化GPIO端口PC6、PC7、PC8
}/*** @brief 错误处理函数* 当系统发生错误时调用该函数。可以在此函数中添加错误处理逻辑,如重启系统或记录日志。*/
void Error_Handler(void)
{__disable_irq(); // 禁用全局中断,防止进一步的干扰while (1) {// 无限循环,防止系统继续运行。可以在此添加LED闪烁等故障指示}
}#ifdef USE_FULL_ASSERT
/*** @brief 断言失败时调用此函数* 当遇到断言参数错误时,报告源文件和行号,帮助调试* @param file 指向发生错误的源文件名的指针* @param line 发生错误的代码行号*/
void assert_failed(uint8_t *file, uint32_t line)
{// 可以在此添加打印或记录错误的实现,例如:// printf("Wrong parameters value: file %s on line %d\r\n", file, line);
}
#endif /* USE_FULL_ASSERT */
ESP8266与STM32
- ESP8266 是一款集成了Wi-Fi通信功能的低功耗微控制器,广泛应用于物联网(IoT)设备中。
- Station 模式:可以像普通的 Wi-Fi 设备一样连接到现有的无线网络。
- AP 模式:可以创建自己的 Wi-Fi 热点,让其他设备连接到 ESP8266。
- Station + Access Point 混合模式(STA + AP 模式):ESP8266 同时充当客户端和热点,既可以连接到现有的 Wi-Fi 网络(作为 STA),又可以作为热点允许其他设备连接到它(作为 AP)。
AT+CWMODE=1:设置为 Station 模式。
AT+CWMODE=2:设置为 AP 模式。
AT+CWMODE=3:设置为 STA + AP 混合模式。
- 服务器
www.daxia.com//下载SSCOM即可
STATION模式
- 配置uart1- printf,uart3-esp上网(因为WiFi芯片在串口3)

使用方法
- 新建stm32工程中添加WiFi-ops.c和WiFi-ops.h
- 开启时钟频率为64Mhz和RCC的Crystal
- 开启UART1和UART3串口为异步通信,且开启中断。
- 并配置LED灯管脚留作测试
- 代码示例
#include "main.h"
#include "stdio.h"
#include "string.h"
#include "wifi-ops.h"// UART句柄
UART_HandleTypeDef huart1;
UART_HandleTypeDef huart3;void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_USART1_UART_Init(void);
static void MX_USART3_UART_Init(void);
// 日志状态标志
int log_success_flags = 0; // 0 - 未初始化,1 - 日志成功,2 - 日志失败/*** @brief ESP8266 Station模式回调函数* @param data: 接收到的数据* @param len: 数据长度* @retval 返回值说明*/
int esp_station_callback(char *data, int len)
{printf("recv from serv: %s\r\n", data);// 更新日志状态标志if (strstr(data, "log success")) {log_success_flags = 1;} else if (strstr(data, "log failed")) {log_success_flags = 2;}return 0;
}/*** @brief UART接收事件回调函数* @param huart: UART句柄* @param Size: 接收的数据大小* @retval None*/
void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{// 处理接收事件,没有传给另一个串口wifi_uart_prepare_idle(huart); // 接收完成后,准备UART进行下一次接收
}/*** @brief 重定向putchar到UART1* @param ch: 要发送的字符* @retval 返回发送的字符*/
int __io_putchar(int ch)
{HAL_UART_Transmit(&huart1, (unsigned char *)&ch, 1, 1);return ch;
}/*** @brief 主函数入口* @retval 程序返回值*/
int main(void)
{HAL_Init(); // 初始化HAL库SystemClock_Config(); // 配置系统时钟// 初始化所有配置的外设MX_GPIO_Init();MX_USART1_UART_Init();MX_USART3_UART_Init();int ret;char sendbuf[128] = {0}; // 用于发送数据的缓冲区// 准备UART3进行ESP8266通信,使用串口3wifi_uart_prepare_idle(&huart3);// 初始化ESP8266 Station模式,使用串口3ret = wifi_station_init(&huart3, esp_station_callback);if (ret < 0) {printf("%s-%d wifi_station_init err\r\n", __func__, __LINE__);return -35;}// 连接到指定的APret = wifi_station_join_ap(&huart3, "xiaomimobile", "12345600");if (ret < 0) {printf("%s-%d wifi_station_join_ap err\r\n", __func__, __LINE__);return -35;}// 连接到指定的TCP服务器ret = wifi_station_tcp_connect(&huart3, "107.148.201.156", 10005);if (ret < 0) {printf("%s-%d wifi_station_tcp_connect err\r\n", __func__, __LINE__);return -35;}// 发送日志请求数据。特定格式snprintf(sendbuf, sizeof(sendbuf), "toServ:action=log,usrname=%s,passwd=%s,devname=sml001;", "xiaowang", "123456");ret = wifi_station_tcp_send_data(&huart3, sendbuf, strlen(sendbuf));if (ret < 0) {printf("%s-%d wifi_station_tcp_send_data err\r\n", __func__, __LINE__);}// 等待日志发送结果for (int i = 0; i < 50; i++) {if (log_success_flags == 2) {printf("%s-%d log failed \r\n", __func__, __LINE__);return -23;} else if (log_success_flags == 1) {printf("%s-%d log success \r\n", __func__, __LINE__);break;}HAL_Delay(10);}// 循环发送获取时间请求,这个是该服务器的特定格式strcpy(sendbuf, "toServ:action=gettime;");while (1){ret = wifi_station_tcp_send_data(&huart3, sendbuf, strlen(sendbuf));if (ret < 0) {printf("%s-%d wifi_station_tcp_send_data err\r\n", __func__, __LINE__);}HAL_Delay(500); // 等待500毫秒}
}/*** @brief 配置系统时钟* @retval None*/
void SystemClock_Config(void)
{RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};// 初始化RCC振荡器RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;RCC_OscInitStruct.HSEState = RCC_HSE_ON;RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;RCC_OscInitStruct.HSIState = RCC_HSI_ON;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL8;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) {Error_Handler();}// 初始化CPU、AHB和APB总线时钟RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK) {Error_Handler();}
}/*** @brief 初始化USART1* @retval None*/
static void MX_USART1_UART_Init(void)
{huart1.Instance = USART1;huart1.Init.BaudRate = 115200;huart1.Init.WordLength = UART_WORDLENGTH_8B;huart1.Init.StopBits = UART_STOPBITS_1;huart1.Init.Parity = UART_PARITY_NONE;huart1.Init.Mode = UART_MODE_TX_RX;huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart1.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart1) != HAL_OK) {Error_Handler();}
}/*** @brief 初始化USART3* @retval None*/
static void MX_USART3_UART_Init(void)
{huart3.Instance = USART3;huart3.Init.BaudRate = 115200;huart3.Init.WordLength = UART_WORDLENGTH_8B;huart3.Init.StopBits = UART_STOPBITS_1;huart3.Init.Parity = UART_PARITY_NONE;huart3.Init.Mode = UART_MODE_TX_RX;huart3.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart3.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart3) != HAL_OK) {Error_Handler();}
}/*** @brief 初始化GPIO* @retval None*/
static void MX_GPIO_Init(void)
{GPIO_InitTypeDef GPIO_InitStruct = {0};// 启用GPIO端口时钟__HAL_RCC_GPIOD_CLK_ENABLE();__HAL_RCC_GPIOB_CLK_ENABLE();__HAL_RCC_GPIOC_CLK_ENABLE();__HAL_RCC_GPIOA_CLK_ENABLE();// 配置GPIO输出级别HAL_GPIO_WritePin(GPIOC, GPIO_PIN_6 | GPIO_PIN_7 | GPIO_PIN_8, GPIO_PIN_RESET);// 配置GPIO引脚GPIO_InitStruct.Pin = GPIO_PIN_6 | GPIO_PIN_7 | GPIO_PIN_8;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(GPIOC, &GPIO_InitStruct);
}/*** @brief 错误处理函数* @retval None*/
void Error_Handler(void)
{__disable_irq();while (1) {}
}/*** @brief 报告断言失败的文件名和行号* @param file: 源文件名* @param line: 错误行号* @retval None*/
void assert_failed(uint8_t *file, uint32_t line)
{// 用户可以添加自己的实现来报告文件名和行号
}
混合模式
- 默认情况下,ESP8266 的 IP 地址在 AP 模式下是 192.168.4.1,以便其他设备连接到该 AP 时可以进行通信。
- 代码
#include "main.h"#include "stdio.h"
#include "string.h"
#include "wifi-ops.h"UART_HandleTypeDef huart1;
UART_HandleTypeDef huart3;void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_USART1_UART_Init(void);
static void MX_USART3_UART_Init(void);#ifdef ESP_STATION/*当stm32收到 esp/服务器发来消息, 会自动执行该函数* data,len 表示收到的具体数据和长度*** 该函数是在 中断执行,不好执行耗时/睡眠工作* */int log_success_flags = 0; //0-uninit 1-log 2-failedint esp_station_callback (char *data,int len)
{printf("recv from serv: %s\r\n",data);//登陆成功失败标志位if( strstr(data,"log success") ){log_success_flags = 1;}else if(strstr( data, "log failed")){log_success_flags = 2;}}
#endif/* 希望手机发来消息,该函数 被调用, data len分别是对方发来的消息和长度*/
int esp_ap_callback(char *data,int len)
{printf("recv from phone: %s\r\n",data); //cmd:wifiname=hyx,wifipasswd=8888888,usrname=xiaowang,passwd=123456;
}void HAL_UARTEx_RxEventCallback(UART_HandleTypeDef *huart, uint16_t Size)
{//other thing to dowifi_uart_prepare_idle(huart); //after recv success, wo must prepare uart for next time
}int __io_putchar(int ch)
{HAL_UART_Transmit(&huart1, (unsigned char *)&ch, 1, 1);return ch;
}int main(void)
{HAL_Init();SystemClock_Config();MX_GPIO_Init();MX_USART1_UART_Init();MX_USART3_UART_Init();/* USER CODE BEGIN 2 */int ret;int linkid; //每个手机都可以连接ap,都会安排一个idchar sendbuf[128]={0};#ifdef ESP_STATION//预处理uart3 for esp ...wifi_uart_prepare_idle(&huart3);ret = wifi_station_init(&huart3, esp_station_callback );if(ret < 0 ){ printf("%s-%d wifi_station_init err\r\n",__func__,__LINE__); return -35;}ret = wifi_station_join_ap(&huart3,"WANGQINGFA","1234567890");if(ret < 0 ){ printf("%s-%d wifi_station_join_ap err\r\n",__func__,__LINE__); return -35;}ret = wifi_station_tcp_connect(&huart3,"107.148.201.156",10001);if(ret < 0 ){ printf("%s-%d wifi_station_tcp_connect err\r\n",__func__,__LINE__); return -35;}/*登陆服务器 toServ:action=log,usrname=xiaowang,passwd=123456,devname=sml001; */snprintf(sendbuf,sizeof(sendbuf), "toServ:action=log,usrname=%s,passwd=%s,devname=sml001;","xiaowang","123456" );ret = wifi_station_tcp_send_data(&huart3,sendbuf,strlen(sendbuf));if(ret < 0 ){ printf("%s-%d wifi_station_tcp_connect err\r\n",__func__,__LINE__); }for(int i=0;i<50;i++){if(log_success_flags == 2){printf("%s-%d log failed \r\n",__func__,__LINE__);return -23;}else if(log_success_flags == 1){printf("%s-%d log success \r\n",__func__,__LINE__);break;}HAL_Delay(10);}
#endifwifi_uart_prepare_idle(&huart3);ret = wifi_ap_init(&huart3, esp_ap_callback);if(ret < 0 ){ printf("%s-%d wifi_ap_init err\r\n",__func__,__LINE__); return -35;}ret = wifi_ap_set_args(&huart3, "zhangsan","12345678");if(ret < 0 ){ printf("%s-%d wifi_ap_set_args err\r\n",__func__,__LINE__); return -35;}linkid = wifi_ap_tcp_listen_and_wait_connect_timeout(&huart3,10001, 5*60*1000);if(linkid < 0 ){ printf("%s-%d wifi_ap_tcp_listen_and_wait_connect_timeout err\r\n",__func__,__LINE__); return -35;}/* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */strcpy(sendbuf,"cmd:hello phone,I have got you messgae,please reset it;");while (1){wifi_ap_tcp_send_data(&huart3, linkid , sendbuf,strlen(sendbuf) );#ifdef ESP_STATIONret = wifi_station_tcp_send_data(&huart3,sendbuf,strlen(sendbuf));if(ret < 0 ){ printf("%s-%d wifi_station_tcp_connect err\r\n",__func__,__LINE__); }
#endifHAL_Delay(500);/* USER CODE END WHILE *//* USER CODE BEGIN 3 */}/* USER CODE END 3 */
}/*** @brief System Clock Configuration* @retval None*/
void SystemClock_Config(void)
{RCC_OscInitTypeDef RCC_OscInitStruct = {0};RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};/** Initializes the RCC Oscillators according to the specified parameters* in the RCC_OscInitTypeDef structure.*/RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;RCC_OscInitStruct.HSEState = RCC_HSE_ON;RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;RCC_OscInitStruct.HSIState = RCC_HSI_ON;RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL8;if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK){Error_Handler();}/** Initializes the CPU, AHB and APB buses clocks*/RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK){Error_Handler();}
}/*** @brief USART1 Initialization Function* @param None* @retval None*/
static void MX_USART1_UART_Init(void)
{/* USER CODE BEGIN USART1_Init 0 *//* USER CODE END USART1_Init 0 *//* USER CODE BEGIN USART1_Init 1 *//* USER CODE END USART1_Init 1 */huart1.Instance = USART1;huart1.Init.BaudRate = 115200;huart1.Init.WordLength = UART_WORDLENGTH_8B;huart1.Init.StopBits = UART_STOPBITS_1;huart1.Init.Parity = UART_PARITY_NONE;huart1.Init.Mode = UART_MODE_TX_RX;huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart1.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart1) != HAL_OK){Error_Handler();}}static void MX_USART3_UART_Init(void)
{huart3.Instance = USART3;huart3.Init.BaudRate = 115200;huart3.Init.WordLength = UART_WORDLENGTH_8B;huart3.Init.StopBits = UART_STOPBITS_1;huart3.Init.Parity = UART_PARITY_NONE;huart3.Init.Mode = UART_MODE_TX_RX;huart3.Init.HwFlowCtl = UART_HWCONTROL_NONE;huart3.Init.OverSampling = UART_OVERSAMPLING_16;if (HAL_UART_Init(&huart3) != HAL_OK){Error_Handler();}/* USER CODE BEGIN USART3_Init 2 *//* USER CODE END USART3_Init 2 */}/*** @brief GPIO Initialization Function* @param None* @retval None*/
static void MX_GPIO_Init(void)
{GPIO_InitTypeDef GPIO_InitStruct = {0};/* GPIO Ports Clock Enable */__HAL_RCC_GPIOD_CLK_ENABLE();__HAL_RCC_GPIOB_CLK_ENABLE();__HAL_RCC_GPIOC_CLK_ENABLE();__HAL_RCC_GPIOA_CLK_ENABLE();/*Configure GPIO pin Output Level */HAL_GPIO_WritePin(GPIOC, GPIO_PIN_6|GPIO_PIN_7|GPIO_PIN_8, GPIO_PIN_RESET);/*Configure GPIO pins : PC6 PC7 PC8 */GPIO_InitStruct.Pin = GPIO_PIN_6|GPIO_PIN_7|GPIO_PIN_8;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(GPIOC, &GPIO_InitStruct);}/* USER CODE BEGIN 4 *//* USER CODE END 4 *//*** @brief This function is executed in case of error occurrence.* @retval None*/
void Error_Handler(void)
{__disable_irq();while (1){}/* USER CODE END Error_Handler_Debug */
}#ifdef USE_FULL_ASSERT
/*** @brief Reports the name of the source file and the source line number* where the assert_param error has occurred.* @param file: pointer to the source file name* @param line: assert_param error line source number* @retval None*/
void assert_failed(uint8_t *file, uint32_t line)
{/* USER CODE BEGIN 6 *//* User can add his own implementation to report the file name and line number,ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) *//* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */
- 如图所示,用的TCP/IP调试软件

相关文章:
STM32与ESP8266的使用
串口透传 “透传”通常指的是数据的透明传输,意思是在不对数据进行任何处理或修改的情况下,将数据从一个接口转发到另一个接口。值得注意的是要避免串口之间无限制的透明,可以采用互斥锁的方式进行限制使用方法 对USART1和USART3(用他俩举例…...
【计算机网络】数据链路层深度解析
概述三个重要问题封装成帧差错检测可靠传输 使用广播信道的数据链路层数据链路层的互连设备 媒体接入MAC地址集线器与交换机区别以太网交换机生成树协议STP 概述 链路就是从一个结点到相邻结点的一段物理线路,而中间没有任何其他的交换结点。数据链路是指把实现通信…...
【基于轻量型架构的WEB开发】【章节作业】
作业1 mybatis核心对象、配置文件和映射文件 一. 单选题(共10题,50分) 1. (单选题)以下关于<select>元素及其属性说法错误的是()。 A. <select>元素用来映射查询语句,它可以帮助我们从数据库中读取出数据,并组装数据给业务开发…...
一张图解析FastAdmin中的表格列表(bootstrap-table)的功能(备份)
功能描述 请根据图片上的数字索引查看对应功能说明。 1.菜单名称和描述 默认生成的CRUD是没有菜单名称和描述显示的,如果需要显示则可以修改权限管理->菜单规则,给对应菜单的添加上备注信息后即可显示,支持HTML 2.TAB过滤选项卡 在一键…...
【数据结构】假设二叉树采用二叉链表存储,编写一棵二又树中序遍历的非递归算法。
编程题: 假设二叉树采用二叉链表存储,编写一棵二又树中序遍历的非递归算法。 分析: 算法描述: 非递归中序遍历二叉树的算法使用栈来辅助实现。首先,从根节点开始,沿着左子树不断向下, 将每个节点压入栈中。当到达最左端节点后,开始出栈并访问节点,接着转向右子树,重…...
李宏毅结构化学习 02
文章目录 一、上篇博文复习二、Separable Case三、Non-separable Case四、Considering Errors五、Regularization六、Structured SVM七、Cutting Plane Algorithm for Structured SVM八、Multi-class and binary SVM九、Beyond Structured SVM 一、上篇博文复习 图中x表示输入的…...
Android AlertDialog圆角背景不生效的问题
一行解决: window?.setBackgroundDrawableResource(android.R.color.transparent) 原文件: /*** Created by Xinghai.Zhao* 自定义选择弹框*/ SuppressLint("InflateParams", "MissingInflatedId") class CustomDialog(context: Context?) : AlertDia…...
探讨基于AI技术的相亲交友系统设计与实现
摘要 随着人工智能技术的发展,相亲交友领域也开始引入AI技术来改善用户体验,提高匹配成功率。本文探讨了如何利用AI技术设计并实现一个智能化的相亲交友系统,该系统能够根据用户的行为数据和个人偏好,自动推荐合适的潜在伴侣。通…...
(2024.9.20)Endnote插入的参考文献字号太大怎么办?
1、序言 常常写论文的人都知道,插入参考文献时,格式调整到让人头大。Endnote的使用大大方便了我们的同时,也意味着我们要学习软件的使用方法。最近重新安装了一下Endnote,插入的文献字体大小就像抽风了一样。在还没有写完文章之前…...
DataGrip在Windows和MacOS平台上的快捷键
0. 背景信息 No.说明1测试DataGrip版本号 : 2024.2.2 1. Windows下快捷键 2. MacOS下快捷键...
CSS---序号使用css设置,counter-reset、counter-increment、content配合实现备注文案的序号展示
直接上代码,全代码copy即可使用! <template><div class"reminder"><span class"Bold_12_body" style"line-height: 8vw">温馨提示:</span><br /><div class"rule-container"…...
Liquor 表达式引擎基本使用
引入依赖 <dependency><groupId>org.noear</groupId><artifactId>liquor-eval</artifactId><version>1.2.7</version> </dependency>liquor 表达式引擎(ExpressionEvaluator)支持 java 所有的类型、及…...
AI美女屠版小红书火了,被当真人推流,颜值博主慌了
最近,微信群里有一条炸裂的聊天记录,传得沸沸扬扬。 聊天记录原主声称,自己通过flux文生图模型跑出AI美女照片,发在小红书上不仅没有被平台标为AI,还成功获得流量扶持。 随后,原主就附上了自己养的1327个小…...
本地搭建我的世界服务器(JAVA)简单记录
网上参考教程挺多的,踩了不少坑,简单记录一下,我做的是一个私人服务器,就是和朋友3、4个人玩。 笨蛋 MC 开服教程 先放一个比较系统和完整的教程,萌新可用,这个教程很详细,我只是记录一下自己的…...
哪个快?用300万个图斑测试ArcGIS Pro的成对叠加与经典叠加
点击下方全系列课程学习 点击学习—>ArcGIS全系列实战视频教程——9个单一课程组合系列直播回放 点击学习——>遥感影像综合处理4大遥感软件ArcGISENVIErdaseCognition 在使用ArcGIS Pro的过程中,很多朋友发现,Pro有个成对叠加工具集。很多…...
超详细!百分百安装成功pytorch,建议收藏
文章目录 一、Anaconda安装1.1下载anaconda1.2配置Anaconda环境1.3验证anaconda是否安装成功 二、查看电脑显卡三、更新显卡驱动3.1下载驱动3.2、查看显卡驱动版本 四、cuda安装4.1CUDA下载4.2CUDA环境配置4.3验证CUDA是否安装成功 五、安装pytorch4.1下载pytorch5.2验证pytorc…...
web基础—dvwa靶场(四)File Inclusion
File Inclusion(文件包含) 有些 web 应用程序允许用户指定直接文件流的输入,或允许用户将文件上载到服务器。稍后 web 应用程序访问 web 应用程序上下文中用户提供的输入。通过这样种操作,web 应用程序允许恶意文件执行。 如果选择要包含的文件是目标计…...
【Python】练习:控制语句(二)第1关
第1关:分支结构基础实训 第一题第二题第三题第四题(※)第五题(※)第六题第七题 第一题 #第一题 for temp in [-280, -100, 0, 20, 120, 200]:#请在下面编写代码# ********** Begin ********** #if temp>-273.15:F9/…...
Vue3 : Pinia的性质与作用
目录 一.性质 二.作用 三.Pinia 的核心概念 四.使用 1.count.ts 2.count.vue Vue 3 中 Pinia 是一个专为 Vue 3 设计的状态管理库,它旨在提供一种简单、直观的方式来管理应用的状态。 一.性质 1.集成性:Pinia 是 Vue 3 官方推荐的状态管理库&…...
对接金蝶云星空调用即时库存信息查询API
文章目录 前言准备工作获取第三方授权权限与授权配置信息集成金蝶云SDK调用实现备注前言 对于有自己商品信息管理后台并且使用金蝶ERP系统管理物料的商家来说,将金蝶上物料的库存信息同步到管理后台就可以不用去金蝶上确认库存了,可以大大简化管理后台的库存变更工作,这篇文…...
IGP(Interior Gateway Protocol,内部网关协议)
IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...
CentOS下的分布式内存计算Spark环境部署
一、Spark 核心架构与应用场景 1.1 分布式计算引擎的核心优势 Spark 是基于内存的分布式计算框架,相比 MapReduce 具有以下核心优势: 内存计算:数据可常驻内存,迭代计算性能提升 10-100 倍(文档段落:3-79…...
在四层代理中还原真实客户端ngx_stream_realip_module
一、模块原理与价值 PROXY Protocol 回溯 第三方负载均衡(如 HAProxy、AWS NLB、阿里 SLB)发起上游连接时,将真实客户端 IP/Port 写入 PROXY Protocol v1/v2 头。Stream 层接收到头部后,ngx_stream_realip_module 从中提取原始信息…...
Fabric V2.5 通用溯源系统——增加图片上传与下载功能
fabric-trace项目在发布一年后,部署量已突破1000次,为支持更多场景,现新增支持图片信息上链,本文对图片上传、下载功能代码进行梳理,包含智能合约、后端、前端部分。 一、智能合约修改 为了增加图片信息上链溯源,需要对底层数据结构进行修改,在此对智能合约中的农产品数…...
技术栈RabbitMq的介绍和使用
目录 1. 什么是消息队列?2. 消息队列的优点3. RabbitMQ 消息队列概述4. RabbitMQ 安装5. Exchange 四种类型5.1 direct 精准匹配5.2 fanout 广播5.3 topic 正则匹配 6. RabbitMQ 队列模式6.1 简单队列模式6.2 工作队列模式6.3 发布/订阅模式6.4 路由模式6.5 主题模式…...
A2A JS SDK 完整教程:快速入门指南
目录 什么是 A2A JS SDK?A2A JS 安装与设置A2A JS 核心概念创建你的第一个 A2A JS 代理A2A JS 服务端开发A2A JS 客户端使用A2A JS 高级特性A2A JS 最佳实践A2A JS 故障排除 什么是 A2A JS SDK? A2A JS SDK 是一个专为 JavaScript/TypeScript 开发者设计的强大库ÿ…...
【从零学习JVM|第三篇】类的生命周期(高频面试题)
前言: 在Java编程中,类的生命周期是指类从被加载到内存中开始,到被卸载出内存为止的整个过程。了解类的生命周期对于理解Java程序的运行机制以及性能优化非常重要。本文会深入探寻类的生命周期,让读者对此有深刻印象。 目录 …...
iview框架主题色的应用
1.下载 less要使用3.0.0以下的版本 npm install less2.7.3 npm install less-loader4.0.52./src/config/theme.js文件 module.exports {yellow: {theme-color: #FDCE04},blue: {theme-color: #547CE7} }在sass中使用theme配置的颜色主题,无需引入,直接可…...
解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist
现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...
uni-app学习笔记三十五--扩展组件的安装和使用
由于内置组件不能满足日常开发需要,uniapp官方也提供了众多的扩展组件供我们使用。由于不是内置组件,需要安装才能使用。 一、安装扩展插件 安装方法: 1.访问uniapp官方文档组件部分:组件使用的入门教程 | uni-app官网 点击左侧…...
