【STM32单片机】----实现LED灯闪烁实战

news2024/12/27 14:08:58

34889d2e7fec4a02bb0ae3a22b011a2a.png

🎩 欢迎来到技术探索的奇幻世界👨‍💻

📜 个人主页:@一伦明悦-CSDN博客

✍🏻 作者简介: C++软件开发、Python机器学习爱好者

🗣️ 互动与支持💬评论      👍🏻点赞      📂收藏     👀关注+

如果文章有所帮助,欢迎留下您宝贵的评论,

点赞加收藏支持我,点击关注,一起进步!

前言

       STM32是一系列由STMicroelectronics开发的32位ARM Cortex-M微控制器系列,广泛应用于嵌入式系统中。它们提供了丰富的外设和性能,适用于各种应用领域,包括工业控制、汽车、消费电子等。STM32系列有多个产品系列,每个系列针对不同的应用需求提供了多种型号和配置选项。要对STM32有一个全面的总结,可以涵盖其主要特点、产品系列、应用领域、开发工具和生态系统等方面的信息。

        代码实现资源链接

【免费】STM32实现LED灯闪烁资源-CSDN文库icon-default.png?t=N7T8https://download.csdn.net/download/m0_59951855/89359786?spm=1001.2014.3001.5503

正文

01-LED灯1实现

       当想要在STM32单片机上实现LED灯的闪烁时,可以遵循以下步骤:

  1. 初始化GPIO引脚:首先,需要初始化用于连接LED的GPIO引脚。这包括设置引脚的模式(输入/输出)、速度、上拉/下拉等。

  2. 配置定时器(Timer):使用一个定时器来生成定期的中断,以便控制LED的闪烁频率。可以选择适当的定时器和预分频器来生成适当的时钟频率。

  3. 编写中断服务程序(ISR):在定时器中断服务程序中,可以切换LED的状态,从而使其闪烁。可以在每次中断时切换LED的状态,或者根据需要计算适当的间隔。

  4. 编写主程序:在主程序中初始化所有必要的硬件和变量,并启动定时器。然后,可以让主程序进入一个无限循环,在循环中等待定时器中断。

  5. 编译和下载程序:最后,将程序编译为可执行文件,并通过调试器将其下载到STM32单片机中进行测试。

        LED灯1实现

        LED1.h文件:

        这段.h文件是LED模块的头文件,用于声明LED相关的函数和变量。让我解释一下:

  1. #ifndef __LED_H 和 #define __LED_H:这是头文件保护措施,确保在同一编译单元中只包含一次该头文件内容,防止重复定义。

  2. void LED1_Init(void);:这是LED1初始化函数的声明,告诉编译器该函数的存在和接口。

  3. #endif:结束头文件的定义。

        这个头文件的作用是在其他源文件中包含它后,可以调用LED1_Init函数进行LED的初始化。

#ifndef __LED_H
#define __LED_H


void LED1_Init(void);

#endif

        LED1.cpp文件:这段代码是用于在STM32F10x系列单片机上初始化一个LED(Light Emitting Diode)的函数。详细解释如下:

  1. #include "stm32f10x.h":这行代码包含了STM32F10x系列的设备头文件,其中包含了该系列单片机的寄存器定义和常量声明等信息。

  2. void LED1_Init(void):这是一个无返回值的函数,用于初始化LED。

  3. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);:这行代码用于使能GPIOA的时钟。在STM32中,访问GPIO需要先使能相应的时钟。

  4. GPIO_InitTypeDef GPIO_InitStructure;:定义了一个结构体变量GPIO_InitStructure,用于配置GPIO初始化参数。

  5. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;:将GPIOA的引脚1配置为推挽输出模式。在这种模式下,GPIO引脚可以提供高电平和低电平输出。

  6. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;:选择了GPIOA的引脚1作为LED连接的引脚。

  7. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;:配置了GPIO的输出速度为50MHz。

  8. GPIO_Init(GPIOA, &GPIO_InitStructure);:根据上述配置初始化GPIOA的引脚1。

  9. 在注释中提到了两种驱动方式:

    • GPIO_ResetBits(GPIOA, GPIO_Pin_1);:将GPIOA的引脚1输出低电平,LED熄灭。
    • GPIO_SetBits(GPIOA, GPIO_Pin_1);:将GPIOA的引脚1输出高电平,LED点亮。

        这段代码的作用是初始化一个LED连接的GPIO引脚,配置为推挽输出模式,并设置输出速度为50MHz。

#include "stm32f10x.h"                  // Device header


void LED1_Init(void)
{
	// 1、开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	// 2、调用Init函数
	// 2-1、需要先进行结构体的定义
	
	GPIO_InitTypeDef GPIO_InitStructure;
	// 数据信号的推挽输出,这个时候STM32对高低电平拥有绝对的控制权,此时LED灯长脚插在PA0口,短脚插在负极,高电平驱动也可以闪烁
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 变成了OD就成了开漏模式,就无法高电平驱动了
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 ;  // 选择Pin_0是因为用的是GPIOA的0号引脚
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;  // 输出速度
	
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	
	// 3、设置高电平,LED灯熄灭
	//GPIO_ResetBits(GPIOA,GPIO_Pin_1);
	//GPIO_SetBits(GPIOA,GPIO_Pin_2);
}


02-LED灯2实现

       LED灯2的实现如下:

        LED2.h文件

        这段.h文件是LED2模块的头文件,用于声明LED2相关的函数和变量。解释如下:

  1. #ifndef __LED2_H 和 #define __LED2_H:这是头文件保护措施,确保在同一编译单元中只包含一次该头文件内容,防止重复定义。

  2. void LED2_Init(void);:这是LED2初始化函数的声明,告诉编译器该函数的存在和接口。

  3. #endif:结束头文件的定义。

        这个头文件的作用是在其他源文件中包含它后,可以调用LED2_Init函数进行LED2的初始化。

#ifndef __LED2_H
#define __LED2_H

void LED2_Init(void);

#endif

        LED2.cpp文件:这段代码是用于在STM32F10x系列单片机上初始化另一个LED的函数。解释如下:

  1. #include "stm32f10x.h":同样是包含了STM32F10x系列的设备头文件,以便使用相关的寄存器定义和常量声明。

  2. void LED2_Init(void):这是另一个无返回值的函数,用于初始化第二个LED。

  3. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);:这行代码用于使能GPIOB的时钟,因为第二个LED连接在GPIOB上。

  4. GPIO_InitTypeDef GPIO_InitStructure;:定义了一个新的结构体变量GPIO_InitStructure,用于配置GPIOB的初始化参数。

  5. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;:同样将GPIOB的引脚1配置为推挽输出模式。

  6. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;:选择了GPIOB的引脚1作为第二个LED连接的引脚。

  7. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;:配置了GPIO的输出速度为50MHz。

  8. GPIO_Init(GPIOB, &GPIO_InitStructure);:根据上述配置初始化GPIOB的引脚1。

        这段代码的作用与之前的LED1初始化函数类似,是初始化另一个LED连接的GPIO引脚,配置为推挽输出模式,并设置输出速度为50MHz。

#include "stm32f10x.h"                  // Device header


void LED2_Init(void)
{
	// 1、开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	// 2、调用Init函数
	// 2-1、需要先进行结构体的定义
	
	GPIO_InitTypeDef GPIO_InitStructure;
	// 数据信号的推挽输出,这个时候STM32对高低电平拥有绝对的控制权,此时LED灯长脚插在PA0口,短脚插在负极,高电平驱动也可以闪烁
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 变成了OD就成了开漏模式,就无法高电平驱动了
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 ;  // 选择Pin_0是因为用的是GPIOA的0号引脚
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;  // 输出速度
	
	GPIO_Init(GPIOB,&GPIO_InitStructure);
}


03-延时函数实现

       Delay的实现如下:

        Delay.h文件

        这个.h文件定义了三个延时函数的声明,分别是Delay_usDelay_msDelay_s,用于提供微秒级、毫秒级和秒级的延时功能。这些声明可以让其他源文件包含这个头文件后直接调用这些函数,而不需要知道函数的具体实现细节。同时,文件开头使用了#ifndef#define宏,以及#endif来实现头文件保护,确保在同一编译单元中只包含一次该头文件内容,防止重复定义。

#ifndef __DELAY_H
#define __DELAY_H

void Delay_us(uint32_t us);  // 微秒延时
void Delay_ms(uint32_t ms);  // 毫秒延时
void Delay_s(uint32_t s);    // 秒延时

#endif

        Delay.cpp文件:这段代码提供了三个延时函数:Delay_usDelay_msDelay_s,分别用于提供微秒级、毫秒级和秒级的延时。下面分别解释这三个函数:

  1. void Delay_us(uint32_t xus):这个函数用于提供微秒级的延时。它使用了STM32的系统滴答(SysTick)定时器来实现延时。xus参数是要延时的微秒数,通过乘以72(STM32的系统时钟频率,HCLK,假设为72MHz)来计算出SysTick定时器的重装载值。然后设置SysTick定时器的时钟源为HCLK,启动定时器,等待定时器计数到0,最后关闭定时器。

  2. void Delay_ms(uint32_t xms):这个函数用于提供毫秒级的延时。它通过调用Delay_us函数来实现,每次调用Delay_us函数延时1000微秒,即1毫秒,通过xms参数指定的次数来控制总的延时毫秒数。

  3. void Delay_s(uint32_t xs):这个函数用于提供秒级的延时。它通过调用Delay_ms函数来实现,每次调用Delay_ms函数延时1000毫秒,即1秒,通过xs参数指定的次数来控制总的延时秒数。

        这三个延时函数都可以在STM32的开发中用于控制程序的执行流程,实现特定的延时效果。

#include "stm32f10x.h"

/**
  * @brief  微秒级延时
  * @param  xus 延时时长,范围:0~233015
  * @retval 无
  */
void Delay_us(uint32_t xus)
{
	SysTick->LOAD = 72 * xus;				//设置定时器重装值
	SysTick->VAL = 0x00;					//清空当前计数值
	SysTick->CTRL = 0x00000005;				//设置时钟源为HCLK,启动定时器
	while(!(SysTick->CTRL & 0x00010000));	//等待计数到0
	SysTick->CTRL = 0x00000004;				//关闭定时器
}

/**
  * @brief  毫秒级延时
  * @param  xms 延时时长,范围:0~4294967295
  * @retval 无
  */
void Delay_ms(uint32_t xms)
{
	while(xms--)
	{
		Delay_us(1000);
	}
}
 
/**
  * @brief  秒级延时
  * @param  xs 延时时长,范围:0~4294967295
  * @retval 无
  */
void Delay_s(uint32_t xs)
{
	while(xs--)
	{
		Delay_ms(1000);
	}
} 

04-主函数实现

       主函数实现代码如下:

        这段主函数代码主要完成了以下几个任务:

  1. 包含了必要的头文件,如STM32F10x系列的设备头文件、延时函数头文件、LED1和LED2的头文件。

  2. 在main函数中调用了LED1_Init和LED2_Init函数,分别初始化了两个LED的引脚。

  3. 进入一个无限循环(while(1)),在循环中实现了LED1和LED2的闪烁效果:

    a. 首先点亮LED1,延时1000ms,然后熄灭LED1,延时1000ms,实现LED1每秒闪烁一次的效果。

    b. 接着点亮LED2,延时500ms,然后熄灭LED2,延时500ms,实现LED2每0.5秒闪烁一次的效果。

        这样,主函数完成了对两个LED灯的初始化和控制,使它们交替闪烁。

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "LED.h"
#include "LED2.h"

int main(void){

	LED1_Init();
	LED2_Init();
	// 4、设置低电平,LED灯亮起
//	GPIO_ResetBits(GPIOA,GPIO_Pin_1);
	// 5、也可以GPIO_WriteBit()函数设置高低电平
	// 前两个参数和Set和Reset一样,第三个参数用于清除端口值,和设置端口值
	// 如果参数=Bit_RESET,清除端口值,设置为低电平,灯亮;反之参数=Bit_SET,为高电平,灯灭
//	GPIO_WriteBit(GPIOA,GPIO_Pin_0,Bit_SET);
	
	// 6、若是需要实现LED灯闪烁的命令,就需要在While死循环中进行一些设置
	while(1){
		// 点亮 两个函数都可以
//		GPIO_ResetBits(GPIOA,GPIO_Pin_0);
//		Delay_ms(500);  // 延时函数直接调用即可
//	    GPIO_WriteBit(GPIOA,GPIO_Pin_0,Bit_RESET);
//		GPIO_WriteBit(GPIOA,GPIO_Pin_0,(BitAction)0);  // 0为低电平
		// 7、这里加延时函数
//		Delay_ms(500);
		
		// 熄灭 
//		GPIO_SetBits(GPIOA,GPIO_Pin_0);
//		Delay_ms(500);
//	    GPIO_WriteBit(GPIOA,GPIO_Pin_0,Bit_SET); 如果这里想要直接使用自己定义的参数代替第三个参数,
//		GPIO_WriteBit(GPIOA,GPIO_Pin_0,(BitAction)1);  // 1为高电平
	//  (BitAction)0 需要加
		
//	    Delay_ms(500);
		GPIO_SetBits(GPIOA,GPIO_Pin_1);
		Delay_ms(1000);
		GPIO_ResetBits(GPIOA,GPIO_Pin_1);
		Delay_ms(1000);
		GPIO_SetBits(GPIOB,GPIO_Pin_1);
		Delay_ms(500);
		GPIO_ResetBits(GPIOB,GPIO_Pin_1);
		Delay_ms(500);
	}

}

总结

       实现LED灯闪烁的核心在于控制GPIO引脚的高低电平状态和添加适当的延时。在STM32上,通过操作寄存器来控制GPIO引脚状态,然后利用延时函数来控制LED的亮灭间隔。

总结如下步骤:

  1. 包含必要的头文件,如STM32的设备头文件和延时函数头文件。

  2. 初始化LED的GPIO引脚,可以通过调用初始化函数实现。

  3. 进入一个无限循环,通常使用while(1)

  4. 在循环中,通过操作GPIO引脚的寄存器,设置LED引脚的状态,从而控制LED的亮灭。

  5. 在LED状态改变后,添加适当的延时,以控制LED的亮灭间隔。

  6. 循环执行步骤4和步骤5,实现LED的闪烁效果。

        这样,通过简单的C语言代码,就可以在STM32上实现LED的闪烁功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1698018.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

只要0.14元带使能控制500mA输出电流3.3V输出电压SOT23-5封装LDO输入电压可达23V的DS8241-33S5

前言: 本来想推荐的,为慎重,搜了下公司信息,改介绍了。公司相关信息在文尾。 该LDO已在立创销售,希望厂家尽早将该款产品做到基础库或免换料费的器件中。 参考价格:0.14元 使能控制:支持 23V …

PCIe总线-事物层之TLP请求和完成报文格式介绍(六)

1.概述 TLP报文按照类型,可以大致分为4中类型,分别是IO请求报文、存储器请求报文、配置请求报文、完成报文和消息请求报文。IO请求报文可分为IO读请求(不携带数据)和IO写请求(携带数据)。存储器请求报文可…

mac 安装Node.js

文章目录 前言一、Node是什么?二、下载三、安装四、验证总结 前言 Node.js是一个开源、跨平台的JavaScript运行时环境,它允许开发者在服务器端运行JavaScript代码。Node.js是基于Chrome V8 JavaScript引擎构建的,它的设计目标是提供一种高效…

番外篇 | YOLOv5-SPD:用最简单的方式完成低分辨率图像和小目标检测升级

前言:Hello大家好,我是小哥谈。论文提出了一个新的CNN构建模块称为SPD-Conv,用来替换每个步长卷转层和每个池化层(从而完全消除它们)。SPD-Conv由一个空间到深度(SPD)层和一个非步长卷积(Conv)层组成。本文详细介绍了如何在YOLOv5中引入SPD-Conv,助力助力低分辨率与小…

pyqt5-结合opencv展示实时视频纯Demo

前言一、代码二、运行结果总结 前言 一、代码 #Author :susocool #Creattime:2024/5/22 #FileName:018.1-显示摄像头 #Description: import sys from PyQt5 import QtWidgets, QtGui, QtCore from PyQt5.QtWidgets import QHBoxLayout import cv2 import osclass V…

k8s集群中pod的容器资源限制和三种探针

一、资源限制 总结: requests表示创建pod时预留的资源,limits表示pod能够使用资源的最大值。requests值可以被超,limits值不能超过,如果是内存使用超过limits会触发oom然后杀掉进程,如果是cpu超过limits会压缩cpu的使用…

Selenium 高频面试题及答案

1、什么是 Selenium?它用于做什么? Selenium 是一个用于自动化测试的开源框架。它提供了多种工具和库,用于模拟用户在不同浏览器和操作系统上的行为,并且可用于测试网页应用程序。 2、Selenium WebDriver 和 Selenium IDE 有何区…

C++学习/复习5--构造函数与初始化/static成员/友元/内部类/匿名对象/编译器的拷贝构造优化

一、本章概要 二、再谈构造函数 1.构造体赋初值与初始化 2.初始化列表与初始化 2.1定义 2.2注意事项与举例 3.explicit关键字与构造函数 3.1隐式类型转换 也叫做自动类型转换 这种转换通常是从存储范围小的类型到存储范围大的类型,或者是从低精度的数值类型到高…

无畏并发: Rust Mutex的基本使用

并发是很多编程语言避不开的一块主要内容,主打一个无畏并发的Rust自然也面临这样的挑战。Rust中的Mutex提供了强大的同步原语,确保共享数据的线程安全,这篇文章中,我们会探讨Mutex的使用,从基础的用法到一些高阶内容。…

基于地理坐标的高阶几何编辑工具算法(8)——整形面

文章目录 工具步骤应用场景算法输入算法输出算法示意图算法原理工具步骤 选中面,点击“整形面”工具,绘制一条和面至少两个交点的线,双击结束。 应用场景 快速修改一个几何面的局部形状。 算法输入 一个待修改的面p,和一条绘制的线l 算法输出 修改后的面 算法示意图…

【JAVA基础之网络编程】UDP和TCP协议以及三次握手和四次挥手的过程

🔥作者主页:小林同学的学习笔录 🔥mysql专栏:小林同学的专栏 目录 1. 网络编程 1.1 概述 1.2 网络编程的三要素 1.2.1 IP地址 1.2.2 InetAddress 1.2.3 端口和协议 1.3 UDP协议 1.3.1 UDP发送数据 1.3.2 UDP接收数据 1.4…

Hadoop伪分布式搭建

1 配置SSH免密登录 1.生成密钥 # ssh-keygen -t rsa 注意:需要经过4次回车 查看密钥及公钥 # cd /root/.ssh 拷贝公钥 # cp id_rsa.pub authorized_keys 2 测试本地免密登录 2 下载Hadoop安装包 使用wget命令从华为云上下载Hadoop安装文件 # wget -P /opt https://m…

数组的理论知识

文章目录 数组的理论知识 数组的理论知识 数组是我们在编程时期经常使用到的一种数据结构。 特点: 在连续的内存空间中存储相同数据类型的数据 如图: arr 数组 注意点:数组的修改的效率是比较慢的,O(n),因为数组只…

6、phpjm混淆解密和php反序列化

题目:青少年雏形系统 1、打开链接也是一个登入面板 2、尝试了sqlmap没头绪 3、尝试御剑,发现一个www.zip 4、下载打开,有一个php文件打开有一段phpjm混淆加密 5、使用手工解混淆 具体解法链接:奇安信攻防社区-phpjm混淆解密浅谈…

国产化服务器开启NTP功能并向NTP时钟服务器同步

1.备份/etc/chrony.conf文件; cp -rp /etc/chrony.conf /etc/chrony.conf.bak.20240522 2.修改chrony.conf文件,增加NTP时钟信息。(客户端填写时钟同步服务器的IP地址或者域名,我这里写的IP地址。下面Allow NTP Client是只允许…

基于YOLOv8+PySide6的快递分类管理系统

1、背景 随着电子商务的飞速发展,快递行业所承受的数据处理需求愈发庞大。在这样的背景下,传统的手工分类方法已经显得力不从心,因其不仅耗时耗力,还存在着易出错的隐患。因此,迫切需要研发出一套高效而准确的自动化系…

Linux VIM指令

三种模式 命令模式:控制屏幕光标的移动,字符、字或行的删除等输入对文件的一些指令 插入模式:对文件内容进行文字输入 底行摸索:文件保存或退出,也可以进行文件替换,找字符串,列出行号等操作…

GAW-1000D 微机控制钢绞线拉力试验机

一、整机外观图与示意图 外观示意图 性能说明: GAW-1000D型微机控制电液伺服钢绞线拉力试验机主要用于对预应力钢绞线进行抗拉强度测试。由宽调速范围的电液比例伺服阀与计算机及测控单元所组成伺服控制系统,能精确的控制和测量试验全过程。整机由主机…

浮点型比较大小

浮点数的存储形式 浮点数按照在内存中所占字节数和数值范围,可以分为浮点型,双精度浮点型和长双浮点型数。 代码: printf("lgn:%e \n", pow(exp(1), 100));printf("lgn:%f ", pow(exp(1), 100));输出结果: …

Mac上如何安装低版本chrome浏览器

背景 为了排查项目上使用chrome低版本的兼容性问题,需要在本机【mac系统】上安装一个低版本的chrome浏览器。 不同版本的chrome下载地址 https://www.slimjet.com/chrome/google-chrome-old-version.php 下载后要记得你下载的旧版本的版本号,后面修改…