STM32单片机外设编程:串口、定时器、中断、ADC的实战应用

发布时间: 2024-07-05 14:48:07 阅读量: 96 订阅数: 39
![STM32单片机外设编程:串口、定时器、中断、ADC的实战应用](https://img-blog.csdnimg.cn/509823d7be834421a341f28adb5146bf.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5aW955qEX-a1qeWQjOWtpg==,size_20,color_FFFFFF,t_70,g_se,x_16) # 1. STM32单片机外设简介** STM32单片机是一款基于ARM Cortex-M内核的32位微控制器,集成了丰富的片上外设,为嵌入式系统开发提供了强大的硬件支持。本章将对STM32单片机的常用外设进行简介,包括串口、定时器、中断、ADC等,为后续章节的实战编程奠定基础。 # 2. 串口编程实战** **2.1 串口通信原理及配置** **2.1.1 串口通信基础** 串口通信是一种异步串行通信协议,用于在两个设备之间传输数据。它使用一条数据线和一条控制线,数据线传输数据位,控制线用于同步通信。串口通信的特点是: * **异步传输:**数据位逐个发送,没有时钟信号同步。 * **半双工通信:**同一时间只能有一个设备发送或接收数据。 * **数据格式:**数据通常以 8 位字节的形式传输,并附加起始位和停止位。 **2.1.2 STM32串口配置** STM32单片机集成了多个串口外设,用于实现串口通信。串口配置主要涉及以下步骤: * **选择串口外设:**STM32单片机有多个串口外设,如 USART1、USART2 等。根据需要选择合适的串口外设。 * **配置波特率:**波特率决定了数据传输速率。通过设置串口外设的波特率寄存器来配置波特率。 * **配置数据格式:**数据格式包括数据位、停止位和奇偶校验位。通过设置串口外设的数据格式寄存器来配置数据格式。 * **配置中断:**串口外设支持中断,当数据接收或发送完成时会触发中断。通过设置串口外设的中断寄存器来配置中断。 **代码块:** ```c #include "stm32f10x.h" void USART1_Init(void) { // 选择串口外设 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE); // 配置波特率 USART_InitTypeDef USART_InitStructure; USART_InitStructure.USART_BaudRate = 115200; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART1, &USART_InitStructure); // 配置中断 USART_ITConfig(USART1, USART_IT_RXNE, ENABLE); NVIC_EnableIRQ(USART1_IRQn); } ``` **逻辑分析:** * `RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);`:使能 USART1 外设时钟。 * `USART_Init(USART1, &USART_InitStructure);`:初始化 USART1 外设,配置波特率、数据格式和模式。 * `USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);`:使能 USART1 接收中断。 * `NVIC_EnableIRQ(USART1_IRQn);`:使能 USART1 中断向量。 **2.2 串口数据收发** **2.2.1 字符串收发** 字符串收发是最常用的串口数据收发方式。通过串口发送和接收字符串,可以实现设备之间的文本信息交换。 **代码块:** ```c void USART1_SendString(char *str) { while (*str) { USART_SendData(USART1, *str); str++; } } char USART1_ReceiveString(void) { char ch; while (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET) ; ch = USART_ReceiveData(USART1); return ch; } ``` **逻辑分析:** * `USART1_SendString(char *str)`:发送一个字符串。循环遍历字符串,逐个字符发送。 * `USART1_ReceiveString(void)`:接收一个字符。等待接收数据寄存器非空,然后读取接收到的字符。 **2.2.2 数据结构收发** 除了字符串,串口还可以收发自定义的数据结构。通过将数据结构序列化为字节流,可以在设备之间传输复杂的数据。 **代码块:** ```c typedef struct { int a; float b; } DataStruct; void USART1_SendDataStruct(DataStruct *data) { USART_SendData(USART1, data->a); USART_SendData(USART1, data->b); } DataStruct USART1_ReceiveDataStruct(void) { DataStruct data; data.a = USART_ReceiveData(USART1); data.b = USART_ReceiveData(USART1); return data; } ``` **逻辑分析:** * `USART1_SendDataStruct(DataStruct *data)`:发送一个数据结构。逐个成员变量发送数据。 * `USART1_ReceiveDataStruct(void)`:接收一个数据结构。逐个成员变量接收数据。 # 3. 定时器编程实战 ### 3.1 定时器工作原理及配置 #### 3.1.1 定时器基础 定时器是一种用于测量时间间隔或生成特定频率脉冲的硬件模块。STM32单片机中有多个定时器,每个定时器都有自己的功能和配置选项。 定时器的工作原理是基于一个计数器,该计数器以特定频率递增。当计数器达到预设值时,它会产生一个中断或触发一个事件。 #### 3.1.2 STM32定时器配置 STM32定时器可以通过寄存器进行配置。主要配置寄存器包括: - **TIMx_CR1:**控制定时器的基本功能,如使能、时钟源、计数方向等。 - **TIMx_PSC:**预分频寄存器,用于设置定时器的时钟分频比。 - **TIMx_ARR:**自动重装载寄存器,用于设置定时器的计数上限。 **示例代码:** ```c // 配置定时器3为向上计数模式,时钟源为APB1,分频比为100 RCC->APB1ENR |= RCC_APB1ENR_TIM3EN; // 使能定时器3时钟 TIM3->CR1 = 0x0000; // 重置定时器3控制寄存器 TIM3->CR1 |= TIM_CR1_CEN; // 使能定时器3 TIM3->PSC = 99; // 设置分频比为100 TIM3->ARR = 10000; // 设置自动重装载值为10000 ``` ### 3.2 定时器中断应用 #### 3.2.1 定时器中断原理 当定时器计数器达到预设值时,会产生一个中断请求。中断请求由NVIC(嵌套向量中断控制器)处理,它会调用相应的中断服务函数。 #### 3.2.2 定时器中断应用示例 定时器中断可以用于各种应用,例如: - **延时:**使用定时器中断实现精确的延时。 - **周期性任务:**定时器中断可以触发周期性任务,如LED闪烁或数据采集。 - **事件计数:**使用定时器中断计数外部事件,如按钮按下次数。 **示例代码:** ```c // 定时器3中断服务函数 void TIM3_IRQHandler(void) { // 清除中断标志位 TIM3->SR &= ~TIM_SR_UIF; // 执行中断处理逻辑 // ... } // 配置定时器3中断 void TIM3_Config_Interrupt(void) { // 使能定时器3中断 TIM3->DIER |= TIM_DIER_UIE; // 设置NVIC中断优先级 NVIC_SetPriority(TIM3_IRQn, 1); // 使能NVIC中断 NVIC_EnableIRQ(TIM3_IRQn); } ``` # 4.1 中断原理及配置 ### 4.1.1 中断基础 中断是一种硬件机制,当发生特定事件时,处理器会暂停当前正在执行的程序,转而执行中断服务程序。中断事件可以由外部设备(如串口、定时器)或内部事件(如错误)触发。 **中断类型** STM32单片机支持多种中断类型,包括: - **外部中断:**由外部引脚上的电平变化触发。 - **内部中断:**由内部外设(如串口、定时器)的事件触发。 - **NMI中断:**由非屏蔽中断请求触发,具有最高优先级。 - **HardFault中断:**由处理器检测到的错误触发,如内存访问违规。 **中断优先级** 每个中断都有一个优先级,决定了当多个中断同时发生时,哪个中断先被处理。STM32单片机支持多级中断优先级,允许用户根据中断的重要性进行配置。 ### 4.1.2 STM32中断配置 **NVIC寄存器** STM32的中断控制器称为NVIC(嵌套向量中断控制器)。它包含一系列寄存器,用于配置和管理中断。 **中断使能寄存器(ISER)** ISER寄存器用于使能或禁用中断。每个中断都有一个对应的位,置1表示使能,置0表示禁用。 **中断挂起寄存器(ISPR)** ISPR寄存器用于挂起中断。置1表示挂起中断,置0表示取消挂起。 **中断优先级寄存器(IPR)** IPR寄存器用于设置中断优先级。每个中断都有一个对应的位,值越大表示优先级越高。 **中断配置步骤** 配置STM32中断的步骤如下: 1. **使能中断:**在ISER寄存器中置1对应中断的位。 2. **设置优先级:**在IPR寄存器中设置对应中断的优先级。 3. **编写中断服务函数:**为每个中断编写对应的中断服务函数。 4. **在中断服务函数中:**处理中断事件,并清除中断标志位。 **代码示例** 以下代码示例展示了如何使能和配置串口中断: ```c // 使能串口中断 NVIC_ISER(NVIC_IRQ_USART1_IRQn) = 1; // 设置串口中断优先级为3 NVIC_IPR(NVIC_IRQ_USART1_IRQn) = 3; ``` # 5. ADC编程实战 ### 5.1 ADC工作原理及配置 #### 5.1.1 ADC基础 模数转换器(ADC)是一种将模拟信号(例如电压或电流)转换为数字信号的电子器件。在STM32单片机中,ADC外设负责执行此转换。 ADC的工作原理是将模拟信号采样并将其转换为数字值。采样频率和分辨率决定了ADC的精度和速度。采样频率是指ADC每秒采样模拟信号的次数,而分辨率是指ADC可以表示的数字值的位数。 #### 5.1.2 STM32 ADC配置 STM32单片机有多个ADC外设,每个外设都有自己的配置寄存器。要配置ADC,需要设置以下参数: - **采样时间:**决定ADC将模拟信号保持多长时间以进行采样。 - **分辨率:**决定ADC可以表示的数字值的位数。 - **通道:**选择要转换的模拟输入通道。 - **触发源:**决定ADC何时开始转换。 ### 5.2 ADC数据采集及处理 #### 5.2.1 ADC数据采集 ADC数据采集过程涉及以下步骤: 1. 配置ADC外设。 2. 触发ADC转换。 3. 读回转换结果。 ```c // ADC配置 ADC_HandleTypeDef hadc1; hadc1.Instance = ADC1; hadc1.Init.ClockPrescaler = ADC_CLOCKPRESCALER_PCLK_DIV2; hadc1.Init.Resolution = ADC_RESOLUTION_12B; hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE; hadc1.Init.ContinuousConvMode = DISABLE; HAL_ADC_Init(&hadc1); // ADC触发转换 HAL_ADC_Start(&hadc1); // 读回转换结果 uint16_t adcValue = HAL_ADC_GetValue(&hadc1); ``` #### 5.2.2 ADC数据处理 ADC数据采集后,通常需要进行处理以提取有用的信息。常见的处理技术包括: - **滤波:**去除ADC数据中的噪声和干扰。 - **校准:**补偿ADC的误差和偏移。 - **单位转换:**将ADC值转换为物理单位(例如电压或电流)。 ```c // ADC数据滤波 uint16_t filteredValue = 0; for (int i = 0; i < 10; i++) { filteredValue += HAL_ADC_GetValue(&hadc1); } filteredValue /= 10; // ADC数据校准 float calibratedValue = filteredValue * 0.985; // ADC数据单位转换 float voltage = calibratedValue * 3.3 / 4095; ``` # 6. 外设综合应用实例 本章将介绍STM32单片机外设的综合应用实例,展示如何将多个外设组合使用,实现更加复杂的功能。 ### 6.1 串口与定时器联动 **应用场景:**定时发送数据到串口 **操作步骤:** 1. 配置定时器,设置中断时间间隔。 2. 在定时器中断服务函数中,发送数据到串口。 ```c // 配置定时器 TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_TimeBaseStructure.TIM_Period = 1000; // 1000ms TIM_TimeBaseStructure.TIM_Prescaler = 7200 - 1; // 1MHz TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); // 配置定时器中断 TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE); // 定时器中断服务函数 void TIM3_IRQHandler(void) { // 清除中断标志位 TIM_ClearITPendingBit(TIM3, TIM_IT_Update); // 发送数据到串口 USART_SendData(USART1, 'A'); } ``` ### 6.2 定时器与中断联动 **应用场景:**定时触发中断 **操作步骤:** 1. 配置定时器,设置中断时间间隔。 2. 在定时器中断服务函数中,执行中断处理逻辑。 ```c // 配置定时器 TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_TimeBaseStructure.TIM_Period = 1000; // 1000ms TIM_TimeBaseStructure.TIM_Prescaler = 7200 - 1; // 1MHz TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); // 配置定时器中断 TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE); // 定时器中断服务函数 void TIM3_IRQHandler(void) { // 清除中断标志位 TIM_ClearITPendingBit(TIM3, TIM_IT_Update); // 中断处理逻辑 // ... } ``` ### 6.3 ADC与串口联动 **应用场景:**采集模拟信号并通过串口发送 **操作步骤:** 1. 配置ADC,设置采样频率和分辨率。 2. 启动ADC转换。 3. 在ADC转换完成中断服务函数中,读取ADC数据并通过串口发送。 ```c // 配置ADC ADC_InitTypeDef ADC_InitStructure; ADC_InitStructure.ADC_Resolution = ADC_Resolution_12b; ADC_InitStructure.ADC_ScanConvMode = DISABLE; ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel = 1; ADC_Init(ADC1, &ADC_InitStructure); // 启动ADC转换 ADC_Cmd(ADC1, ENABLE); // ADC转换完成中断服务函数 void ADC1_IRQHandler(void) { // 清除中断标志位 ADC_ClearITPendingBit(ADC1, ADC_IT_EOC); // 读取ADC数据 uint16_t adc_data = ADC_GetConversionValue(ADC1); // 通过串口发送ADC数据 USART_SendData(USART1, (adc_data >> 8) & 0xFF); USART_SendData(USART1, adc_data & 0xFF); } ```
corwn 最低0.47元/天 解锁专栏
送3个月
profit 百万级 高质量VIP文章无限畅学
profit 千万级 优质资源任意下载
profit C知道 免费提问 ( 生成式Al产品 )

相关推荐

Big黄勇

硬件工程师
广州大学计算机硕士,硬件开发资深技术专家,拥有超过10多年的工作经验。曾就职于全球知名的大型科技公司,担任硬件工程师一职。任职期间负责产品的整体架构设计、电路设计、原型制作和测试验证工作。对硬件开发领域有着深入的理解和独到的见解。
专栏简介
《STM32单片机程序教程》专栏是STM32单片机开发人员的宝贵资源。它涵盖了从入门到高级的广泛主题,包括编程入门、DMA、CAN总线、USB通信、项目实战、调试技巧、优化编程、应用案例、开发环境配置、库函数详解、固件升级、安全编程、故障分析、性能优化、多任务编程、图形界面开发、传感器编程、电机控制、图像处理和人工智能应用。无论您是初学者还是经验丰富的开发人员,本专栏都提供了全面而深入的指导,帮助您掌握STM32单片机的各个方面,并将其应用于各种实际项目中。
最低0.47元/天 解锁专栏
送3个月
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
C知道 免费提问 ( 生成式Al产品 )

最新推荐

【Python集合异常处理攻略】:集合在错误控制中的有效策略

![【Python集合异常处理攻略】:集合在错误控制中的有效策略](https://blog.finxter.com/wp-content/uploads/2021/02/set-1-1024x576.jpg) # 1. Python集合的基础知识 Python集合是一种无序的、不重复的数据结构,提供了丰富的操作用于处理数据集合。集合(set)与列表(list)、元组(tuple)、字典(dict)一样,是Python中的内置数据类型之一。它擅长于去除重复元素并进行成员关系测试,是进行集合操作和数学集合运算的理想选择。 集合的基础操作包括创建集合、添加元素、删除元素、成员测试和集合之间的运

Python版本与性能优化:选择合适版本的5个关键因素

![Python版本与性能优化:选择合适版本的5个关键因素](https://ask.qcloudimg.com/http-save/yehe-1754229/nf4n36558s.jpeg) # 1. Python版本选择的重要性 Python是不断发展的编程语言,每个新版本都会带来改进和新特性。选择合适的Python版本至关重要,因为不同的项目对语言特性的需求差异较大,错误的版本选择可能会导致不必要的兼容性问题、性能瓶颈甚至项目失败。本章将深入探讨Python版本选择的重要性,为读者提供选择和评估Python版本的决策依据。 Python的版本更新速度和特性变化需要开发者们保持敏锐的洞

Python print语句装饰器魔法:代码复用与增强的终极指南

![python print](https://blog.finxter.com/wp-content/uploads/2020/08/printwithoutnewline-1024x576.jpg) # 1. Python print语句基础 ## 1.1 print函数的基本用法 Python中的`print`函数是最基本的输出工具,几乎所有程序员都曾频繁地使用它来查看变量值或调试程序。以下是一个简单的例子来说明`print`的基本用法: ```python print("Hello, World!") ``` 这个简单的语句会输出字符串到标准输出,即你的控制台或终端。`prin

【Python字典的并发控制】:确保数据一致性的锁机制,专家级别的并发解决方案

![【Python字典的并发控制】:确保数据一致性的锁机制,专家级别的并发解决方案](https://media.geeksforgeeks.org/wp-content/uploads/20211109175603/PythonDatabaseTutorial.png) # 1. Python字典并发控制基础 在本章节中,我们将探索Python字典并发控制的基础知识,这是在多线程环境中处理共享数据时必须掌握的重要概念。我们将从了解为什么需要并发控制开始,然后逐步深入到Python字典操作的线程安全问题,最后介绍一些基本的并发控制机制。 ## 1.1 并发控制的重要性 在多线程程序设计中

Python数组在科学计算中的高级技巧:专家分享

![Python数组在科学计算中的高级技巧:专家分享](https://media.geeksforgeeks.org/wp-content/uploads/20230824164516/1.png) # 1. Python数组基础及其在科学计算中的角色 数据是科学研究和工程应用中的核心要素,而数组作为处理大量数据的主要工具,在Python科学计算中占据着举足轻重的地位。在本章中,我们将从Python基础出发,逐步介绍数组的概念、类型,以及在科学计算中扮演的重要角色。 ## 1.1 Python数组的基本概念 数组是同类型元素的有序集合,相较于Python的列表,数组在内存中连续存储,允

Python函数调用栈分析:追踪执行流程,优化函数性能的6个技巧

![function in python](https://blog.finxter.com/wp-content/uploads/2021/02/round-1024x576.jpg) # 1. 函数调用栈基础 函数调用栈是程序执行过程中用来管理函数调用关系的一种数据结构,它类似于一叠盘子的堆栈,记录了程序从开始运行到当前时刻所有函数调用的序列。理解调用栈对于任何希望深入研究编程语言内部运行机制的开发者来说都是至关重要的,它能帮助你解决函数调用顺序混乱、内存泄漏以及性能优化等问题。 ## 1.1 什么是调用栈 调用栈是一个后进先出(LIFO)的栈结构,用于记录函数调用的顺序和执行环境。

Python装饰模式实现:类设计中的可插拔功能扩展指南

![python class](https://i.stechies.com/1123x517/userfiles/images/Python-Classes-Instances.png) # 1. Python装饰模式概述 装饰模式(Decorator Pattern)是一种结构型设计模式,它允许动态地添加或修改对象的行为。在Python中,由于其灵活性和动态语言特性,装饰模式得到了广泛的应用。装饰模式通过使用“装饰者”(Decorator)来包裹真实的对象,以此来为原始对象添加新的功能或改变其行为,而不需要修改原始对象的代码。本章将简要介绍Python中装饰模式的概念及其重要性,为理解后

Python pip性能提升之道

![Python pip性能提升之道](https://cdn.activestate.com/wp-content/uploads/2020/08/Python-dependencies-tutorial.png) # 1. Python pip工具概述 Python开发者几乎每天都会与pip打交道,它是Python包的安装和管理工具,使得安装第三方库变得像“pip install 包名”一样简单。本章将带你进入pip的世界,从其功能特性到安装方法,再到对常见问题的解答,我们一步步深入了解这一Python生态系统中不可或缺的工具。 首先,pip是一个全称“Pip Installs Pac

【递归与迭代决策指南】:如何在Python中选择正确的循环类型

# 1. 递归与迭代概念解析 ## 1.1 基本定义与区别 递归和迭代是算法设计中常见的两种方法,用于解决可以分解为更小、更相似问题的计算任务。**递归**是一种自引用的方法,通过函数调用自身来解决问题,它将问题简化为规模更小的子问题。而**迭代**则是通过重复应用一系列操作来达到解决问题的目的,通常使用循环结构实现。 ## 1.2 应用场景 递归算法在需要进行多级逻辑处理时特别有用,例如树的遍历和分治算法。迭代则在数据集合的处理中更为常见,如排序算法和简单的计数任务。理解这两种方法的区别对于选择最合适的算法至关重要,尤其是在关注性能和资源消耗时。 ## 1.3 逻辑结构对比 递归

Pandas中的文本数据处理:字符串操作与正则表达式的高级应用

![Pandas中的文本数据处理:字符串操作与正则表达式的高级应用](https://www.sharpsightlabs.com/wp-content/uploads/2021/09/pandas-replace_simple-dataframe-example.png) # 1. Pandas文本数据处理概览 Pandas库不仅在数据清洗、数据处理领域享有盛誉,而且在文本数据处理方面也有着独特的优势。在本章中,我们将介绍Pandas处理文本数据的核心概念和基础应用。通过Pandas,我们可以轻松地对数据集中的文本进行各种形式的操作,比如提取信息、转换格式、数据清洗等。 我们会从基础的字
最低0.47元/天 解锁专栏
送3个月
百万级 高质量VIP文章无限畅学
千万级 优质资源任意下载
C知道 免费提问 ( 生成式Al产品 )