STM32串口通信(发送与接收数据)

2024-03-01 04:20

本文主要是介绍STM32串口通信(发送与接收数据),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 一、介绍部分
    • 通信接口
      • 术语解释
    • 串口通信简介
    • 硬件电路
    • 电平标准
    • 串口参数
    • 串口时序
    • USART简介
      • USART框图
      • USRAT基本结构
      • 数据帧
      • 起始位检测
      • 波特率发生器
      • CH340G
  • 二、实例部分
    • 使用串口发送数据
    • 接线图
      • 代码实现
      • 重定向printf需要勾上Use MicroLIB
      • 中文不乱码方法
    • 串口的发送与接收数据
      • 线路连接与上面一致
      • 代码实现


前言

串口通信(Serial Communications)的概念非常简单,串口按位(bit)发送和接收字节。尽管比按字节(byte)的并行通信慢,但是串口可以在使用一根线发送数据的同时用另一根线接收数据。本文主要接收使用串口发送接收数据,波特率设置,串口的基本时序等。


一、介绍部分

通信接口

在这里插入图片描述

术语解释

在这里插入图片描述

串口通信简介

在这里插入图片描述

硬件电路

在这里插入图片描述

电平标准

在这里插入图片描述

串口参数

在这里插入图片描述

串口时序

在这里插入图片描述

USART简介

在这里插入图片描述

USART框图

在这里插入图片描述

USRAT基本结构

在这里插入图片描述

数据帧

在这里插入图片描述
在这里插入图片描述

起始位检测

在受到噪声影响后,采用2:1策略,选择更多的作为所接收到的数据,并使噪声标志位NE置1

在这里插入图片描述
在这里插入图片描述

波特率发生器

在这里插入图片描述
在这里插入图片描述

CH340G

在这里插入图片描述

二、实例部分

使用USART1来作为例子,根据引脚定义,选择正确的接口
在这里插入图片描述

使用串口发送数据

接线图

在这里插入图片描述

代码实现

配置串口Serial.c

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>void Serial_Init(void){// 开启时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);// 初始化引脚,发送数据引脚GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;		// 复用推挽输出GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;	// A9GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		// 50Hz翻转速度GPIO_Init(GPIOA, &GPIO_InitStructure);// 初始化串口配置USART_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate = 9600; // 串口波特率USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; // 不使用流控USART_InitStructure.USART_Mode = USART_Mode_Tx; // 串口模式,发送USART_InitStructure.USART_Parity = USART_Parity_No; // 无校验USART_InitStructure.USART_StopBits = USART_StopBits_1; // 选择一位停止位USART_InitStructure.USART_WordLength = USART_WordLength_8b; // 不需要校验位,八位字长USART_Init(USART1,&USART_InitStructure);// USART1使能USART_Cmd(USART1,ENABLE);
}// 发送函数
void USART_SendByte(uint8_t Byte){USART_SendData(USART1,Byte);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);
}// 发送数组函数
void USART_SendArray(uint8_t *Array,uint16_t Length){uint8_t i = 0;for(i=0;i<Length;i++){USART_SendData(USART1,Array[i]);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);}
}// 发送字符串函数
void USART_SendString(uint8_t *String){uint8_t i = 0;for(i=0;String[i]!='\0';i++){USART_SendData(USART1,String[i]);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);}
}// 返回X的Y次方
uint32_t Serial_Pow(uint32_t X,uint32_t Y){uint32_t Result = 1;while(Y--){Result *= X;}return Result;
}
// 发送数字函数
void USART_SendNum(uint32_t Num,uint16_t Length){uint8_t i = 0;for(i=0;i<Length;i++){USART_SendByte(Num / Serial_Pow(10,Length-i-1) % 10 + 0x30);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);}
}//重定向fputc函数,fputc是printf函数的底层,printf通过不停的调用fputc来达到输出的效果
//重定向到串口
int fputc(int ch,FILE *f){USART_SendByte(ch);return ch;
}// 封装使用sprintf输出到串口
void Serial_Printf(char *format, ...)
{char String[100];va_list arg;							// 可变参数列表va_start(arg, format);		// 从format开始接收可变参数vsprintf(String, format, arg);va_end(arg);USART_SendString((uint8_t*)String);
}

主函数main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Serial.h"int main(void)
{OLED_Init();Serial_Init();//USART_SendByte();
//	uint8_t Array[] = {0x41,0x42,0x43,0x44};
//	USART_SendArray(Array,4);
//	uint8_t String[] = {"hello world"};
//	USART_SendString(String);
//	USART_SendNum((uint32_t)12345,5);
//	printf("Num = %d\r\n",666);
//	char String[100];
//	sprintf(String,"Num = %d\r\n",666);
//	USART_SendString((uint8_t*)String);Serial_Printf("一程山水");while (1){}
}

重定向printf需要勾上Use MicroLIB

在这里插入图片描述

中文不乱码方法

  1. 代码与串口都使用utf8格式,并在如下图位置加上–no-multibyte-chars
  2. 使用GB2312支持中文编码格式,串口使用GBK编码格式接收即可。

串口的发送与接收数据

线路连接与上面一致

代码实现

串口配置Serial.c

#include "stm32f10x.h"                  // Device header
#include <stdio.h>
#include <stdarg.h>uint8_t RxData;
uint8_t RxFlag;void Serial_Init(void){// 开启时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);// 初始化引脚GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;		// 复用推挽输出GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;	// A9 发送数据GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		// 50Hz翻转速度GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;		// 上拉输出GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;	// A10 接收数据GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		// 50Hz翻转速度GPIO_Init(GPIOA, &GPIO_InitStructure);// 初始化串口配置USART_InitTypeDef USART_InitStructure;USART_InitStructure.USART_BaudRate = 9600; // 串口波特率USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; // 不使用流控USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx; // 串口模式,发送+接收USART_InitStructure.USART_Parity = USART_Parity_No; // 无校验USART_InitStructure.USART_StopBits = USART_StopBits_1; // 选择一位停止位USART_InitStructure.USART_WordLength = USART_WordLength_8b; // 不需要校验位,八位字长USART_Init(USART1,&USART_InitStructure);// 开启中断USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);//初始化NVICNVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);	// 分组NVIC_InitTypeDef NVIC_InitStructure;// 中断通道NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;// 中断通道使能NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;// 抢占优先级NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;// 响应优先级NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;NVIC_Init(&NVIC_InitStructure);// USART1使能USART_Cmd(USART1,ENABLE);
}// 发送函数
void USART_SendByte(uint8_t Byte){USART_SendData(USART1,Byte);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);
}// 发送数组函数
void USART_SendArray(uint8_t *Array,uint16_t Length){uint8_t i = 0;for(i=0;i<Length;i++){USART_SendData(USART1,Array[i]);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);}
}// 发送字符串函数
void USART_SendString(uint8_t *String){uint8_t i = 0;for(i=0;String[i]!='\0';i++){USART_SendData(USART1,String[i]);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);}
}// 返回X的Y次方
uint32_t Serial_Pow(uint32_t X,uint32_t Y){uint32_t Result = 1;while(Y--){Result *= X;}return Result;
}
// 发送数字函数
void USART_SendNum(uint32_t Num,uint16_t Length){uint8_t i = 0;for(i=0;i<Length;i++){USART_SendByte(Num / Serial_Pow(10,Length-i-1) % 10 + 0x30);// 等待写入完成,写入完成之后会将标志位自动清0while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);}
}//重定向fputc函数,fputc是printf函数的底层,printf通过不停的调用fputc来达到输出的效果
//重定向到串口
int fputc(int ch,FILE *f){USART_SendByte(ch);return ch;
}// 封装使用sprintf输出到串口
void Serial_Printf(char *format, ...)
{char String[100];va_list arg;							// 可变参数列表va_start(arg, format);		// 从format开始接收可变参数vsprintf(String, format, arg);va_end(arg);USART_SendString((uint8_t*)String);
}// 获取RxFlag
uint8_t USART_GetRxFlag(void){if(RxFlag == 1){RxFlag = 0;return 1;}return 0;
}// 获取RxData
uint8_t USART_GetRxData(void){return RxData;
}//中断函数
void USART1_IRQHandler(void){if(USART_GetITStatus(USART1,USART_IT_RXNE)==SET){RxData = USART_ReceiveData(USART1);RxFlag = 1;USART_ClearITPendingBit(USART1,USART_IT_RXNE);}
}

主函数main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "Serial.h"uint8_t Serial_RxData;int main(void)
{OLED_Init();OLED_ShowString(1, 1, "RxData:");Serial_Init();while (1){if (USART_GetRxFlag() == 1){Serial_RxData = USART_GetRxData();USART_SendByte(Serial_RxData);OLED_ShowHexNum(1, 8, Serial_RxData, 2);}}
}

这篇关于STM32串口通信(发送与接收数据)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/761198

相关文章

Django中使用SMTP实现邮件发送功能

《Django中使用SMTP实现邮件发送功能》在Django中使用SMTP发送邮件是一个常见的需求,通常用于发送用户注册确认邮件、密码重置邮件等,下面我们来看看如何在Django中配置S... 目录1. 配置 Django 项目以使用 SMTP2. 创建 Django 应用3. 添加应用到项目设置4. 创建

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

【STM32】SPI通信-软件与硬件读写SPI

SPI通信-软件与硬件读写SPI 软件SPI一、SPI通信协议1、SPI通信2、硬件电路3、移位示意图4、SPI时序基本单元(1)开始通信和结束通信(2)模式0---用的最多(3)模式1(4)模式2(5)模式3 5、SPI时序(1)写使能(2)指定地址写(3)指定地址读 二、W25Q64模块介绍1、W25Q64简介2、硬件电路3、W25Q64框图4、Flash操作注意事项软件SPI读写W2

vue2 组件通信

props + emits props:用于接收父组件传递给子组件的数据。可以定义期望从父组件接收的数据结构和类型。‘子组件不可更改该数据’emits:用于定义组件可以向父组件发出的事件。这允许父组件监听子组件的事件并作出响应。(比如数据更新) props检查属性 属性名类型描述默认值typeFunction指定 prop 应该是什么类型,如 String, Number, Boolean,

STM32(十一):ADC数模转换器实验

AD单通道: 1.RCC开启GPIO和ADC时钟。配置ADCCLK分频器。 2.配置GPIO,把GPIO配置成模拟输入的模式。 3.配置多路开关,把左面通道接入到右面规则组列表里。 4.配置ADC转换器, 包括AD转换器和AD数据寄存器。单次转换,连续转换;扫描、非扫描;有几个通道,触发源是什么,数据对齐是左对齐还是右对齐。 5.ADC_CMD 开启ADC。 void RCC_AD

STM32内部闪存FLASH(内部ROM)、IAP

1 FLASH简介  1 利用程序存储器的剩余空间来保存掉电不丢失的用户数据 2 通过在程序中编程(IAP)实现程序的自我更新 (OTA) 3在线编程(ICP把整个程序都更新掉) 1 系统的Bootloader写死了,只能用串口下载到指定的位置,启动方式也不方便需要配置BOOT引脚触发启动  4 IAP(自己写的Bootloader,实现程序升级) 1 比如蓝牙转串口,

FreeRTOS-基本介绍和移植STM32

FreeRTOS-基本介绍和STM32移植 一、裸机开发和操作系统开发介绍二、任务调度和任务状态介绍2.1 任务调度2.1.1 抢占式调度2.1.2 时间片调度 2.2 任务状态 三、FreeRTOS源码和移植STM323.1 FreeRTOS源码3.2 FreeRTOS移植STM323.2.1 代码移植3.2.2 时钟中断配置 一、裸机开发和操作系统开发介绍 裸机:前后台系

寻迹模块TCRT5000的应用原理和功能实现(基于STM32)

目录 概述 1 认识TCRT5000 1.1 模块介绍 1.2 电气特性 2 系统应用 2.1 系统架构 2.2 STM32Cube创建工程 3 功能实现 3.1 代码实现 3.2 源代码文件 4 功能测试 4.1 检测黑线状态 4.2 未检测黑线状态 概述 本文主要介绍TCRT5000模块的使用原理,包括该模块的硬件实现方式,电路实现原理,还使用STM32类

linux中使用rust语言在不同进程之间通信

第一种:使用mmap映射相同文件 fn main() {let pid = std::process::id();println!(

STM32 ADC+DMA导致写FLASH失败

最近用STM32G070系列的ADC+DMA采样时,遇到了一些小坑记录一下; 一、ADC+DMA采样时进入死循环; 解决方法:ADC-dma死循环问题_stm32 adc dma死机-CSDN博客 将ADC的DMA中断调整为最高,且增大ADCHAL_ADC_Start_DMA(&hadc1, (uint32_t*)adc_buffer, ADC_Buffer_Size); 的ADC_Bu