【STM32】基于HAL库实现流水灯及中断控制


前言
本次实验的器件及软件如下:
软件版本:
STM32CubeMX:6.35,KEIl:5.31,FlyMcu或者mcuisp(烧入软件)
硬件:
STM32F103C8T 核心板,面包板,usb to ttl ,杜邦线若干,LED灯三个

一、通过STMCube配置项目并实现LED流水灯

1、项目配置

1、基于芯片开始项目的创建在这里插入图片描述
2、注意输入芯片型号
在这里插入图片描述
3、配置调试接口
在这里插入图片描述
4、配置时钟模式,具体如下
在这里插入图片描述
Disable 内部时钟
BYPASS:旁路时钟
Crystal:晶振/陶瓷振荡器

4、配置GPIO
这里我们需要三个输出口,简单点,就选选用PA0,PA1,PA2,三个引脚
点击要用的引脚,然后设置,注意三个都要设置
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
以上两张图来自资料,这个实验中我们可以忽略这些

5、时钟配置,
具体配置如下图
在这里插入图片描述
6、工程配置
点击Project manager
自己命名工程名,不要出现空格,选择路径,最好自己新建一个文件夹
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
到此我们就成功创建完一个工程

2、LED流水灯的实现

1、首先我们回到keil
找到main.c文件
在这里插入图片描述

在main函数的while循环中写入以下代码
在这里插入图片描述

                  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_RESET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
				  HAL_Delay(1000);
				
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_RESET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
				  HAL_Delay(1000);
		
			  	HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
			   	HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_RESET);
			   	HAL_Delay(1000);

然后点击build

2、接着我们烧入程序
要注意程序文件的选择,电平的设置
先点击读器件信息,然后点击开始编程

在这里插入图片描述
一些线路的连接可参考以下链接,链接: link 注意自己选的引脚

效果图如下:

二、中断控制实现LED流水灯

1、什么是中断,中断有什么作用

中断是指CPU在执行当前程序时系统出现了某种状况,使得CPU必须停止当前程序,而去执行另一段程序来处理的出现的紧急事务,处理结束后CPU再返回到原先暂停的程序继续执行,这个过程就称为中断。
使得计算机系统具备应对对处理突发事件的能力,使其能及时响应紧急事件。
提高处理器效率,如果没有中断系统,CPU就只能按照原来的程序编写的先后顺序,对各个外设进行查询和处理,即轮询工作方式,轮询方法貌似公平,但实际工作效率却很低。
这里仅做简单介绍,具体的可自行查找。

2、项目创建

在第一个LED流水灯的基础上,我们还需要进行以下配置
1、除以上三个引脚,我们还需要在选一个引脚,作为中断的触发这里选用PB5
具体操作如下图:

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

2、时钟配置
在这里插入图片描述
3、NVIC配置如下图;
在这里插入图片描述
在这里插入图片描述

然后就行生成代码步骤即可

3、代码实现

1、在main函数上写入

uint32_t sign=0;//自定义中断标识符号
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){
  if(GPIO_Pin == SWITCH_Pin){
      //获取B5的电位
     GPIO_PinState pinState = HAL_GPIO_ReadPin(SWITCH_GPIO_Port,SWITCH_Pin);
 
      //高电位
      if(pinState==GPIO_PIN_SET)//PB5高电位流水灯亮
      {
		  sign=1;
	  }
      //低电位
      else if(pinState==GPIO_PIN_RESET)//PB5低电位流水灯灭
	  {
		  sign=0;//PB5低电位标识
	  }
      } 
   }

在这里插入图片描述

2、在while循环中

switch(sign)
		{
			case 0:
			{
					HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_RESET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
				  HAL_Delay(500);
				if(sign==1)
				{	
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
				   break;
				}
				
			  	HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_RESET);
			   	HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
			   	HAL_Delay(500);
				if(sign==1)
				{	
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
				   break;
				}
				
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
				  HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_RESET);
				  HAL_Delay(500);
			}
			
		
			case 1:
		
			{
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_0,GPIO_PIN_SET);
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_1,GPIO_PIN_SET);
				HAL_GPIO_WritePin(GPIOA,GPIO_PIN_2,GPIO_PIN_SET);
				break;
			}

		}

在这里插入图片描述

4、线路连接及效果演示

1、线路连接
在这里插入图片描述

PA0----红
PA1----黄
PA2----绿

PB5做为中断的触发。
实物连接如下图:

在这里插入图片描述

2、效果图如下

流水灯及中断

在这里插入图片描述

三、实验心得

本次实验让我初步的了解了hal库的使用,并初步学习了用STM32CubeMX创建相关的工程的流程,实际上和基于寄存器,基于标准库的开发方式有相通之处,熟悉的掌握hal库的使用,可以有效的提高我们的开发效率,减少代码的书写。同时初步了解了中断。

  • 2
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,以下是基于STM32CubeIDE和HAL库的定时器中断实现LED流水灯的代码: ```c #include "main.h" #include "stm32f4xx_hal.h" /* 定义GPIO引脚 */ #define LED1_Pin GPIO_PIN_0 #define LED1_GPIO_Port GPIOA #define LED2_Pin GPIO_PIN_1 #define LED2_GPIO_Port GPIOA #define LED3_Pin GPIO_PIN_2 #define LED3_GPIO_Port GPIOA #define LED4_Pin GPIO_PIN_3 #define LED4_GPIO_Port GPIOA /* 定义全局变量和函数 */ TIM_HandleTypeDef htim2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_TIM2_Init(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_TIM2_Init(); HAL_TIM_Base_Start_IT(&htim2); /* 启动定时器并开启中断 */ while (1) { /* 主函数不做任何事情 */ } } /* 定时器中断处理函数 */ void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { static uint8_t led_status = 0x01; /* 初始状态下第一个LED亮 */ /* 根据led_status位控制LED的亮灭 */ if (led_status & 0x01) HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED1_GPIO_Port, LED1_Pin, GPIO_PIN_RESET); if (led_status & 0x02) HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED2_GPIO_Port, LED2_Pin, GPIO_PIN_RESET); if (led_status & 0x04) HAL_GPIO_WritePin(LED3_GPIO_Port, LED3_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED3_GPIO_Port, LED3_Pin, GPIO_PIN_RESET); if (led_status & 0x08) HAL_GPIO_WritePin(LED4_GPIO_Port, LED4_Pin, GPIO_PIN_SET); else HAL_GPIO_WritePin(LED4_GPIO_Port, LED4_Pin, GPIO_PIN_RESET); /* 更新led_status的值 */ if (led_status == 0x08) led_status = 0x01; else led_status <<= 1; } /* System Clock Configuration */ void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Configure the main internal regulator output voltage */ __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; 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_HSI; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK) { Error_Handler(); } } /* GPIO初始化 */ void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* 使能GPIOA时钟 */ __HAL_RCC_GPIOA_CLK_ENABLE(); /* 配置GPIO引脚为输出模式 */ GPIO_InitStruct.Pin = LED1_Pin | LED2_Pin | LED3_Pin | LED4_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } /* 定时器初始化 */ void MX_TIM2_Init(void) { /* 使能TIM2时钟 */ __HAL_RCC_TIM2_CLK_ENABLE(); /* 初始化htim2的各项参数 */ htim2.Instance = TIM2; htim2.Init.Prescaler = 8399; /* 预分频值 */ htim2.Init.CounterMode = TIM_COUNTERMODE_UP; /* 向上计数模式 */ htim2.Init.Period = 999; /* 自动重装值 */ htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; /* 时钟分频 */ if (HAL_TIM_Base_Init(&htim2) != HAL_OK) { Error_Handler(); } } ``` 注释已经非常详细了,简单来说就是通过定时器中断不断地改变LED的状态,从而实现LED流水灯的效果。每当定时器计数器达到自动重装值时,就会产生一次定时器中断,然后在中断处理函数中改变LED的状态。 需要注意的是,这里使用的定时器是TIM2,预分频值为8399,自动重装值为999,因此定时器的计数频率为84MHz / (8399 + 1) = 10kHz,即每隔100ms产生一次定时器中断。如果需要改变LED流水灯的速度,可以调整预分频值和自动重装值。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值