STM32——led灯的点亮+闪烁+流水灯的实现

一、LED点亮

1.原理

其方式有两种一种是寄存器方式一种是库函数方式,但其原理都是一样的。如原理图所示,与LED相连接的IO口置低电平即可点亮led灯
在这里插入图片描述

2.寄存器方式–代码

	//寄存器方式
	//主频8Mhz的时候实现1us延时的方法
	//LED灯
	//1,开时钟PE时钟
	RCC->APB2ENR |= 1<<6;
	//2,配置IO口
	GPIOE->CRL &= ~(0xffff<<8);//先清零指定寄存器位
	GPIOE->CRL |= 0x1111<<8;//设置模式为推挽输出模式
	//3, 设置开关灯
	GPIOE->ODR |= 0xf<<2;//关灯操作

3.库函数–代码

(1)led.c:函数led_config()

	//库函数
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE,ENABLE);	
	GPIO_InitTypeDef GPIO_InitStruct={0};
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_Out_PP;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_2;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_2MHz;
	GPIO_Init(GPIOE,&GPIO_InitStruct);
	GPIO_ResetBits(GPIOE,GPIO_Pin_2);

(2)main.c调用led_config

4.结果展示

在这里插入图片描述

二、LED闪烁

原理:让连接的IO口先置零再置1可以使其先亮再灭,我们先使灯置1,然后宏定义让灯反转,delay一个时间,代码如下
(1)led.c函数led_config()置1:

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE,ENABLE);	
	GPIO_InitTypeDef GPIO_InitStruct={0};
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_Out_PP;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_2;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_2MHz;
	GPIO_Init(GPIOE,&GPIO_InitStruct);
	GPIO_SetBits(GPIOE,GPIO_Pin_2);

(2)led.h宏定义灯的反转

#define LED1_Toggle() GPIOE->ODR ^= (1<<2)//反转灯操作

(3)main.c调用

int main(void){
LED_Config();
while(1)
{
		if(sys_time>=1000)
		{
			sys_time = 0;
			LED1_Toggle();
		}
}

sys_time在delay.c中定义

#include "delay.h"

//主频72Mhz的时候实现1us延时的方法
#define delay_1us()   __nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();\
__nop();__nop();
void Delay_us(uint32_t time)
{
	while(time--)
	{
		delay_1us();
	}
}
void Delay_ms(uint32_t time)
{
	while(time--)
	{
		Delay_us(1000);
	}
}

uint32_t led_time = 0;
uint32_t sys_time = 0;
uint32_t dht11_time = 0;

void SysTick_Handler(void)//1ms进入一次
{
	led_time++;
	sys_time++;
	dht11_time++;
}

void Sys_ms(uint32_t time)
{
	uint32_t temp = time+sys_time;
	while(temp>sys_time);
}


delay.h

#ifndef __DELAY_H
#define __DELAY_H

#include "stm32f10x.h"
//extern uint16_t temperature;
//extern uint16_t humidity;
extern uint32_t led_time ;
extern uint32_t sys_time ;
extern uint32_t dht11_time ;
void Delay_us(uint32_t time);
void Delay_ms(uint32_t time);

#endif

ps:systick_handler是由系统定时器(SysTick)触发的中断处理函数。SysTick是一种基于硬件的定时器,它可以在每个时钟周期结束时触发中断。在ARM Cortex-M系列处理器中,SysTick定时器是由内核提供的,并且可以用于实现操作系统的时间片轮转调度、延时等功能。当SysTick定时器计数器减到时,就会触发systick_handler中断处理函数。因此,systick_handler的触发是由SysTick定时器的计数器减到时自动触发的。

三、LED流水灯

使这四个灯轮流亮起-代码如下

1.代码

int main(void){
LED_Config();
while(1)
{
		if(sys_time>=1000)
		{
			sys_time = 0;
			LED1_Toggle();
			Delay_ms(300);
			LED2_Toggle();
			Delay_ms(300);
			LED3_Toggle();
			Delay_ms(300);
			LED4_Toggle();
			Delay_ms(300);
		}
}

2.结果展示

流水灯

对于在STM32实现流水灯效果,可以使用定时器来控制LED闪烁频率。以下是一个基本的流水灯示例程序,使用了TIM2定时器来控制LED闪烁频率: ```c #include "stm32f4xx.h" void GPIO_Init(void) { RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOD, ENABLE); GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_100MHz; GPIO_InitStruct.GPIO_OType = GPIO_OType_PP; GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_Init(GPIOD, &GPIO_InitStruct); } void TIM2_Init(void) { RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); TIM_TimeBaseInitTypeDef TIM_InitStruct; TIM_InitStruct.TIM_Prescaler = 41999; // 设置预分频值,得到1ms的定时周期 TIM_InitStruct.TIM_CounterMode = TIM_CounterMode_Up; TIM_InitStruct.TIM_Period = 999; // 设置计数器最大值,得到1s的定时周期 TIM_InitStruct.TIM_ClockDivision = 0; TIM_InitStruct.TIM_RepetitionCounter = 0; TIM_TimeBaseInit(TIM2, &TIM_InitStruct); TIM_Cmd(TIM2, ENABLE); } void TIM2_IRQHandler(void) { if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { static uint8_t ledIndex = 0; static uint16_t ledValue = GPIO_Pin_12; GPIO_SetBits(GPIOD, ledValue); if (ledIndex == 0) { GPIO_ResetBits(GPIOD, GPIO_Pin_15); ledValue = GPIO_Pin_12; } else { GPIO_ResetBits(GPIOD, ledValue >> 1); ledValue <<= 1; } if (++ledIndex > 3) { ledIndex = 0; } TIM_ClearITPendingBit(TIM2, TIM_IT_Update); } } int main(void) { GPIO_Init(); TIM2_Init(); NVIC_InitTypeDef NVIC_InitStruct; NVIC_InitStruct.NVIC_IRQChannel = TIM2_IRQn; NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0; NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStruct); TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); while (1) { } } ``` 上述代码中,我们首先初始化了GPIOD的引脚作为输出,用于连接LED。然后,我们初始化了TIM2定时器,设置了预分频和计数器最大值,使得定时器的计数周期为1ms。在定时器中断处理函数中,控制LED闪烁效果。最后,在主函数中使能了TIM2的中断,并进入了一个无限循环。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值