第六届 蓝桥杯 嵌入式 省赛

news2025/4/23 3:51:55

参考

第六届蓝桥杯嵌入式省赛程序设计题解析(基于HAL库)_蓝桥杯嵌入式第六届真题-CSDN博客

一、分析功能

RTC 定时

1)时间初始化

2)定时上报电压时间

ADC测量

采集电位器的输出电压信号。

串行功能

1)传送要设置的k值

2)传送上报的电压

3)保存 E2PROM

LED指示灯

电压大于阈值时,闪烁。

LCD显示

1)电位器输出电压

2)k 值

3)LED闪烁开关状态

4)系统时间

按键

1)开关LED闪烁

2)设置上报时间

3)切换时、分、秒

4)调整时间

整体逻辑图

二、CubeMX 配置

1.基础配置

新建一个工程,选择芯片 STM32G431RBT6

 配置系统 SYS

修改中断优先级

配置时钟RCC

时钟树设置: 

输入频率其实就是晶振提供的24MHz,采用HSE(高速外部时钟源)。至于SYSCLK 系统时钟的80MHz 可以通过 PLLM、PLL配置出来。

2. KEY+LED

 

设置电气属性

将 PC8-PC15 以及 PD2 设置为输出;

将 PB0,PB1,PB2,PA0 设置为 输入。

设置 LED 的初始状态

由于LED低有效,故设置LED的输出设置成High-----这样上电的时候灯是熄灭的

3. UART

设置MODE为异步通信(Asynchronous)
修改波特率9600(根据题目修改)
NVIC Settings一栏使能接受中断

4. ADC

5. TIM

使能定时器6(基本定时器) 

配置定时器2(输入捕获)

PA15选择定时器2的通道1

 

配置定时器3(输出PWM)

选择80分频,一兆的频率进行1000计数,频率就是1000HZ,使能自动重装载。

TIM3的占空比设置成30%,则脉冲就设置成300 

TIM17的占空比设置成60%,则脉冲设置成600

配置定时器15(输出方波---是比较输出模式)

6. RTC

RTC时钟频率 = RTC时钟源 / ((Asynchronous Predivider value + 1) * (Synchronous Predivider value + 1))

7. 生成代码

三、编写代码

mcu 开发的架构层次:

由于蓝桥杯是没有 RTOS 的,且BSP和HAL都是现成的。只要开发应用层和修改MCU即可即可。

接下来是对应用层的开发。

1. 头文件

mcu编程和系统编程的代码风格有点不一样。mcu貌似把头文件全放main.c上了,而系统编程都是放在 *.h 中。可能mcu的代码量不够多,没必要这么写。

Cube设定好后,会有这些应用层的文件:

也就是 ADC、RTC、UART、TIM。

LED和KEY是不用再另起一个文件写,但是为了书写规范,还是打算写一个key_led.c。

而LCD是需要从赛点资源库导入进来的。

/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "adc.h"
#include "rtc.h"
#include "tim.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "lcd.h"
#include "i2c_hal.h"

2. 函数声明和主函数

函数申明

void Key_Proc(void);
void Led_Proc(void);
void Lcd_Proc(void);
void Usart_Proc(void);

主函数

int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  MX_ADC2_Init();
  MX_TIM2_Init();
  MX_TIM6_Init();
  MX_USART1_UART_Init();
  MX_RTC_Init();
  MX_TIM3_Init();
  MX_TIM15_Init();
  MX_TIM17_Init();
  /* USER CODE BEGIN 2 */
	LCD_Init();
  LCD_Clear(White);
  LCD_SetBackColor(White);
  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */
		Key_Proc();
		Led_Proc();
		Lcd_Proc();
		Usart_Proc();
    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

3. 函数定义

3.1 按键、LCD

我们重新审题:

B1:LED灯的打开和关闭

B2:LCD 的切换

B3:切换时分秒

B4:修改时分秒

时间窗口
//*减速变量
//方式1
__IO uint32_t uwTick_Key_Set_Point = 0;//控制Key_Proc的执行速度


if((uwTick -  uwTick_Key_Set_Point)<50)	return;//减速函数
		uwTick_Key_Set_Point = uwTick;

//方式2
uint32_t uskey;

//stm32g4xx_it.c
/* USER CODE BEGIN PV */
extern uint32_t uskey;
/* USER CODE END PV */
void SysTick_Handler(void)
{
/* USER CODE BEGIN SysTick_IRQn 0 */
/* USER CODE END SysTick_IRQn 0 */
HAL_IncTick();
/* USER CODE BEGIN SysTick_IRQn 1 */
uskey++;//实现usled每隔1ms自增1
/* USER CODE END SysTick_IRQn 1 */
}

两种写法,挑一个。

按键模版
//*按键扫描专用变量
uint8_t ucKey_Val, unKey_Down, ucKey_Up, ucKey_Old;

ucKey_Val = Key_Scan();
unKey_Down = ucKey_Val & (ucKey_Old ^ ucKey_Val); 
ucKey_Up = ~ucKey_Val & (ucKey_Old ^ ucKey_Val);	
ucKey_Old = ucKey_Val;

典型的3行按键模版,背就完了。

uint8_t Key_Scan(void);

uint8_t Key_Scan(void)
{
	uint8_t key_val=0;
	if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_0)==GPIO_PIN_RESET)
	{
		key_val=1;
	}
	if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_1)==GPIO_PIN_RESET)
	{
		key_val=2;
	}
	if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_2)==GPIO_PIN_RESET)
	{
		key_val=3;
	}
	if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==GPIO_PIN_RESET)
	{
		key_val=4;
	}
	return key_val;
}
B1按键功能实现
void Key_Proc(void)
{
	if((uwTick -  uwTick_Key_Set_Point)<50)	return;//减速函数
		uwTick_Key_Set_Point = uwTick;
	
	ucKey_Val = Key_Scan();
	unKey_Down = ucKey_Val & (ucKey_Old ^ ucKey_Val); 
	ucKey_Up = ~ucKey_Val & (ucKey_Old ^ ucKey_Val);	
	ucKey_Old = ucKey_Val;

	//B1完成LED报警功能的打开和关闭
	if(unKey_Down == 1)
	{
		LED_Ctrl ^= 0x01;//让最后一位翻滚
	}
}

对LED异或1是切换状态的基本操作。

LCD显示

LCD的切换,就是清屏后,再刷上去。

LCD 的两个界面:

界面1

1、电位器输出电压

2、k值

3、LED的状态

4、系统时间

界面2

1、Setting 

2、XX-XX-XX

延时后,读取电压和RTC,而电压与RTC的实现后面会实现。 

__IO uint32_t uwTick_Lcd_Set_Point = 0;//控制Lcd_Proc的执行速度
float R37_Voltage;
RTC_TimeTypeDef H_M_S_Time;
RTC_DateTypeDef Y_M_D_Date;

void Lcd_Proc(void)
{
	if((uwTick -  uwTick_Lcd_Set_Point)<100)	return;//减速函数
		uwTick_Lcd_Set_Point = uwTick;
	
	//数据采集区
	R37_Voltage = ((((float)getADC2())/4096)*3.3);
	HAL_RTC_GetTime(&hrtc, &H_M_S_Time, RTC_FORMAT_BIN);//读取日期和时间必须同时使用
	HAL_RTC_GetDate(&hrtc, &Y_M_D_Date, RTC_FORMAT_BIN);
	
}

 那么我们要设置两个page

page0:数据显示区
uint8_t Interface_Num;//0x00-显示界面,0x10-设置上报时间的小时,0x11-设置分钟,0x12-设置秒。
uint8_t Lcd_Disp_String[21];//最多显示20个字符
uint8_t k_int = 1;

void Lcd_Proc(void)
{
    ......
	
	if(Interface_Num == 0x00)	//page0
	{
		sprintf((char *)Lcd_Disp_String, " V1:%4.2fV", R37_Voltage);
		LCD_DisplayStringLine(Line2, Lcd_Disp_String);
		
		sprintf((char *)Lcd_Disp_String, " k:%3.1f", (k_int*0.1));
		LCD_DisplayStringLine(Line4, Lcd_Disp_String);
		
		if(LED_Ctrl == 0)
			sprintf((char *)Lcd_Disp_String, "     LED:ON");
		else
			sprintf((char *)Lcd_Disp_String, "     LED:OFF");
		LCD_DisplayStringLine(Line6, Lcd_Disp_String);
		
		sprintf((char *)Lcd_Disp_String, "     T:%02d-%02d-%02d", (unsigned int)H_M_S_Time.Minutes, (unsigned int)H_M_S_Time.Second;
		LCD_DisplayStringLine(Line8, Lcd_Disp_String);
	}
}

UART传输的数据,后面会实现。

page1:时间设置区

设置的 Interface_Num 为 0x10,以及右移4位,能够兼顾 0x10,0x11,0x12 三个状态保持在同Page 中。

__IO uint32_t uwTick_SETTING_TIME_Set_Point = 0;//控制待设置的时间数值闪烁
uint8_t SETTING_TIME_Ctrl = 0;// 0-亮,1-灭,控制时间设置界面的待设置值的闪烁功能

void Lcd_Proc(void)
{
    ......
	
	//时间设置区
	if((Interface_Num>>4) == 0x1)	//进入设置界面
	{
		sprintf((char *)Lcd_Disp_String, "     Setting");
		LCD_DisplayStringLine(Line2, Lcd_Disp_String);
		sprintf((char *)Lcd_Disp_String, "     T:%02d-%02d-%02d", (unsigned int)Clock_Comp_Disp[0], (unsigned int)Clock_Comp_Disp[1], (unsigned int)Clock_Comp_Disp[2]);
		
		if((uwTick - uwTick_SETTING_TIME_Set_Point)>=500)
		{
			uwTick_SETTING_TIME_Set_Point = uwTick;
			SETTING_TIME_Ctrl ^= 0x1;
		}
		
		if(SETTING_TIME_Ctrl == 0x1)	//控制闪烁,时间设置的时候闪烁
		{
			if(Interface_Num == 0x10)	//设置时
			{
				Lcd_Disp_String[6] = ' ';
				Lcd_Disp_String[7] = ' ';
			}
			else if(Interface_Num == 0x11)	//设置分
			{
				Lcd_Disp_String[9] = ' ';
				Lcd_Disp_String[10] = ' ';
			}
			else if(Interface_Num == 0x12)	//设置秒
			{
				Lcd_Disp_String[12] = ' ';
				Lcd_Disp_String[13] = ' ';
			}
		}
		LCD_DisplayStringLine(Line5, Lcd_Disp_String);
	}
}

RTC的时间后面会实现。

B2按键功能实现
uint8_t Clock_Comp_Disp[3] = {0,0,0};//闹钟比较值的初值(显示专用)
uint8_t Clock_Comp_Ctrl[3] = {0,0,0};//闹钟比较值的初值(控制专用)

void Key_Proc(void)
{
    ......	
	//B2完成两个界面的切换
	if(unKey_Down == 2)
	{
		if(Interface_Num == 0x00)	//数据显示区
		{
			LCD_Clear(White);	//清屏
			Interface_Num = 0x10;
		}
		else if((Interface_Num>>4) == 0x1)
		{
			LCD_Clear(White);	//清屏
			Interface_Num = 0x00;
			
			Clock_Comp_Ctrl[0] = Clock_Comp_Disp[0];	//更新闹钟显示值到控制值
			Clock_Comp_Ctrl[1] = Clock_Comp_Disp[1];	//更新闹钟显示值到控制值	
			Clock_Comp_Ctrl[2] = Clock_Comp_Disp[2];	//更新闹钟显示值到控制值	
		}
	}
}

B3按键功能实现
void Key_Proc(void)
{
    ...
	//B3切换时分秒,切换时会闪烁
	if(unKey_Down == 3)
	{
		if((Interface_Num>>4) == 0x1)
			//时分秒循环切换
			if(++Interface_Num == 0x13)
				Interface_Num = 0x10;
	}
}

B4功能实现

时间设定,具体是在 LCD 部分实现。按键只实现时间重置而已。

设置的 Interface_Num 为 0x10,以及右移4位,能够兼顾 0x10,0x11,0x12 三个状态保持在同Page 中。

void Key_Proc(void)
{
    ......

	//B4调整时间,其实按键这块只是实现时间到底后回到0
	if(unKey_Down == 4)
	{
		if(Interface_Num == 0x10)
		{
			if( ++Clock_Comp_Disp[0] ==24)
				Clock_Comp_Disp[0] = 0;
		}
		if(Interface_Num == 0x11)
		{
			if( ++Clock_Comp_Disp[1] ==60)
				Clock_Comp_Disp[1] = 0;
		}
		if(Interface_Num == 0x12)
		{
			if( ++Clock_Comp_Disp[2] ==60)
				Clock_Comp_Disp[2] = 0;
		}
	}
}

3.2 LED

重新审题

V 1 >V DD *k 时,指示灯LD1 0.2 秒为间隔闪烁,闪烁功能可以通过按键关闭。
__IO uint32_t uwTick_Led_Set_Point = 0;//控制Led_Proc的执行速度
uint8_t ucLed;
__IO uint32_t uwTick_LED_bulingbuling_Set_Point = 0;//控制LED报警闪烁的打点变量


void Led_Proc(void){
	if((uwTick -  uwTick_Led_Set_Point)<50)	return;//减速函数
		uwTick_Led_Set_Point = uwTick;
	if(LED_Ctrl == 0x1)//关闭LED的功能的时候
		ucLed = 0x00;
	else//开启LED功能的时候
	{
		if(R37_Voltage>=(3.3*k_int*0.1))
		{
			//200ms 闪烁
			if((uwTick-uwTick_LED_bulingbuling_Set_Point)>=200)
			{
				uwTick_LED_bulingbuling_Set_Point = uwTick;
				ucLed ^= 0x1;
			}
		}
		else
			ucLed = 0x00;
	}
	LED_Disp(ucLed);
}
void LED_Disp(uint8_t ucLed);
void LED_Disp(uint8_t ucLed)
{
	//**将所有的灯熄灭
	HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15|GPIO_PIN_8
												|GPIO_PIN_9|GPIO_PIN_10|GPIO_PIN_11|GPIO_PIN_12, GPIO_PIN_SET);
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_SET);		
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_RESET);
 
	//根据ucLed的数值点亮相应的灯
	HAL_GPIO_WritePin(GPIOC, ucLed<<8, GPIO_PIN_RESET);
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_SET);		
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_RESET);	
}

3.3 串口

重新审题

定时上报电压V1

格式:V1电压值】+k 值】+【时间】【命令结束标志】

“2.21+0.1+123030\n”
12 30 30 秒上报电压值为 2.21V k 值为 0.1
串口接收数据
当第一个字符收到 k,则开始缓存。
uint8_t rx_buffer;
uint8_t rx_buf[100];//接收到的指令临时存放缓冲区
uint8_t rx_buf_index = 0;//控制数据往buf里边存储的顺序
_Bool Start_Flag;//起始位判断

void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
 
	if((rx_buffer == 0x6B)&&(rx_buf_index == 0))
	{
		Uart_Rev_Data_Delay_Time = uwTick;//接收到第一个数据启动计时		
		Start_Flag = 1;
	}
	if(Start_Flag == 1)
	{		
		rx_buf[rx_buf_index] = rx_buffer;
		rx_buf_index++;		
	}	
	HAL_UART_Receive_IT(&huart1, (uint8_t *)(&rx_buffer), 1);	
}
串口数据处理
uint8_t Ctrl_Uart_Send_Time_Data_Times = 0;// 控制只允许到闹钟时间后只上报一次
__IO uint32_t Uart_Rev_Data_Delay_Time = 0;//控制串口接收数据的等待时间
_Bool Start_Flag;//起始位判断


uint16_t counter = 0;
uint8_t str[40];
uint8_t rx_buffer;
uint8_t rx_buf[100];//接收到的指令临时存放缓冲区
uint8_t rx_buf_index = 0;//控制数据往buf里边存储的顺序

void Usart_Proc(void){
	if((uwTick -  uwTick_Usart_Set_Point)<30)	return;//减速函数
		uwTick_Usart_Set_Point = uwTick;

	//闹钟时间到
	if((H_M_S_Time.Hours == Clock_Comp_Ctrl[0]&&(H_M_S_Time.Minutes == Clock_Comp_Ctrl[1]&&(H_M_S_Time.Seconds == Clock_Comp_Ctrl[2]))))
	{
		//控制只发送一次数据
		if(Ctrl_Uart_Send_Time_Data_Times == 0)
		{
			Ctrl_Uart_Send_Time_Data_Times = 1;
			sprintf(str, "%4.2f+%3.1f+%02d%02d%02d\n", R37_Voltage, (k_int*0.1), (unsigned int)H_M_S_Time.Hour, (unsigned int)H_M_S_Time.Minute, (unsigned int)H_M_S_Time.Second);
			HAL_UART_Transmit(&huart1, (unsigned char *) str, strlen(strlen), 50);
		}
	}
	else
		//当时间变化或者控制值变化,两者不等的时候,恢复下一次数据发送允许。
		Ctrl_Uart_Send_Time_Data_Times = 0;
	
	//串口接收的数据处理
	if(((uwTick - Uart_Rev_Data_Delay_Time)<=300)&&(uwTick - Uart_Rev_Data_Delay_Time)>=200)//200ms~300ms之内处理数据
	{
		if(rx_buf_index == 6)	//接收到6个数据
		{
			if((rx_buf[0] == 0x6B)&&(rx_buf[1] == 0x30)&&(rx_buf[2] == 0x2E)&&(rx_buf[4] == 0x5C)&&(rx_buf[5] == 0x6E))
			{
				if((rx_buf[3]>=0x31)&&(rx_buf[3]<=0x39))
				{
					k_int = rx_buf[3] - 0x30;
					sprintf(str, "OK\n");
					HAL_UART_Transmit(&huart1, (unsigned char *) strlen, strlen(strlen), 50);
					iic_24c02_write(&k_int, 0, 1);
				}
			}
		}
	rx_buf_index = 0;
	Start_Flag = 0;
	}
}
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
 
	if((rx_buffer == 0x6B)&&(rx_buf_index == 0))
	{
		Uart_Rev_Data_Delay_Time = uwTick;//接收到第一个数据启动计时		
		Start_Flag = 1;
	}
	if(Start_Flag == 1)
	{		
		rx_buf[rx_buf_index] = rx_buffer;
		rx_buf_index++;		
	}	
	HAL_UART_Receive_IT(&huart1, (uint8_t *)(&rx_buffer), 1);	
}

发送数据没啥好说的,而接收 "k0.x\n" 需要注意,总共就6个元素:

  • 检查数据是否符合特定格式:

    • 0x6B 对应ASCII字符 'k'

    • 0x30 对应ASCII字符 '0'

    • 0x2E 对应ASCII字符 '.'

    • 0x5C 对应ASCII字符 ''

    • 0x6E 对应ASCII字符 'n'

除了 buf[3],其余都要检查。

最后 使用 iic_24c02_write 将k的数值写到k_int。

3.4 RTC

在前面,我已经定义过这两个变量了,这里稍微再提一下。

RTC_TimeTypeDef H_M_S_Time;
RTC_DateTypeDef Y_M_D_Date;

HAL_RTC_GetTime(&hrtc,&time,RTC_FORMAT_BIN);
HAL_RTC_GetDate(&hrtc,&date,RTC_FORMAT_BIN);

3.5 ADC

uint16_t getADC1(void)
{
	uint16_t adc = 0;
	
	HAL_ADC_Start(&hadc1);
	adc = HAL_ADC_GetValue(&hadc1);
	
	return adc;
}
uint16_t getADC2(void)
{
	uint16_t adc = 0;
	
	HAL_ADC_Start(&hadc2);
	adc = HAL_ADC_GetValue(&hadc2);
	
	return adc;
}

3.6 RCC

这部分,没有什么需要用户自己编写的地方

3.7 I2C

void iic_24c02_write(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum)
{
	I2CStart();
	I2CSendByte(0xa0);
	I2CWaitAck();
	
	I2CSendByte(ucAddr);
	I2CWaitAck();
	
	while(ucNum--)
	{
		I2CSendByte(*pucBuf++);
		I2CWaitAck();
	}
	I2CStop();
	HAL_Delay(500);
}
void iic_24c02_read(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum)
{
	I2CStart();
	I2CSendByte(0xa0);
	I2CWaitAck();
	I2CSendByte(ucAddr);
	I2CWaitAck();
	I2CStart();
	I2CSendByte(0xa1);
	I2CWaitAck();
	while(ucNum--)
	{
		*pucBuf++=I2CReceiveByte();
		if(ucNum)
			I2CSendAck();
		else
			I2CSendNotAck();
	}
	I2CStop();
}

4. 代码规范

4.1 MAIN

main.c 头文件

系统文件和Cube自动配置的:

/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "adc.h"
#include "rtc.h"
#include "tim.h"
#include "usart.h"
#include "gpio.h"
#include <stdio.h>   // 提供 sprintf()
#include <string.h>  // 提供 strlen()

第三方导入文件:

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "lcd.h"
#include "i2c_hal.h"
/* USER CODE END Includes */

结构体变量

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
RTC_TimeTypeDef H_M_S_Time;
RTC_DateTypeDef Y_M_D_Date;

/* USER CODE END PTD */

用户定义的全局变量

//*减速变量
__IO uint32_t uwTick_Key_Set_Point = 0;//控制Key_Proc的执行速度
__IO uint32_t uwTick_Led_Set_Point = 0;//控制Led_Proc的执行速度
__IO uint32_t uwTick_Lcd_Set_Point = 0;//控制Lcd_Proc的执行速度
__IO uint32_t uwTick_Usart_Set_Point = 0;//控制Usart_Proc的执行速度
 
//*按键扫描专用变量
uint8_t ucKey_Val, unKey_Down, ucKey_Up, ucKey_Old;
 
//*LED专用变量
uint8_t ucLed;
 
//*LCD显示专用变量
uint8_t Lcd_Disp_String[21];//最多显示20个字符
 
//*串口专用变量
uint16_t counter = 0;
char str[40];
uint8_t rx_buffer;
uint8_t rx_buf[100];//接收到的指令临时存放缓冲区
uint8_t rx_buf_index = 0;//控制数据往buf里边存储的顺序
 
//用户自定义变量区
uint8_t Interface_Num;//0x00-显示界面,0x10-设置上报时间的小时,0x11-设置分钟,0x12-设置秒。
float R37_Voltage;
uint8_t k_int = 1;
uint8_t LED_Ctrl = 0;// 0-打开,1关闭,控制LED报警功能
uint8_t Clock_Comp_Disp[3] = {0,0,0};//闹钟比较值的初值(显示专用)
uint8_t Clock_Comp_Ctrl[3] = {0,0,0};//闹钟比较值的初值(控制专用)
__IO uint32_t uwTick_SETTING_TIME_Set_Point = 0;//控制待设置的时间数值闪烁
uint8_t SETTING_TIME_Ctrl = 0;// 0-亮,1-灭,控制时间设置界面的待设置值的闪烁功能
uint8_t Ctrl_Uart_Send_Time_Data_Times = 0;// 控制只允许到闹钟时间后只上报一次
__IO uint32_t Uart_Rev_Data_Delay_Time = 0;//控制串口接收数据的等待时间
_Bool Start_Flag;//起始位判断
__IO uint32_t uwTick_LED_bulingbuling_Set_Point = 0;//控制LED报警闪烁的打点变量

函数声明

/* USER CODE BEGIN PFP */
void Key_Proc(void);
void Led_Proc(void);
void Lcd_Proc(void);
void Usart_Proc(void);

/*这部分封装到BSP中*/
//uint8_t Key_Scan(void);
//void LED_Disp(uint8_t ucLed);
//void iic_24c02_write(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum);

/* USER CODE END PFP */

不过,我更习惯将函数声明放在 main.h 中。

4.2 BSP

gpio

LED的 void LED_Disp(uint8_t ucLed) 和 KEY的 uint8_t Key_Scan(void) 可以封装到 gpio.c 中。

//gpio.h
/* USER CODE BEGIN Prototypes */
uint8_t Key_Scan(void);
void LED_Disp(uint8_t ucLed);
/* USER CODE END Prototypes */


//gpio.c
/* USER CODE BEGIN 2 */
//LED扫描
void LED_Disp(uint8_t ucLed)
{
	//**将所有的灯熄灭
	HAL_GPIO_WritePin(GPIOC, GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15|GPIO_PIN_8
												|GPIO_PIN_9|GPIO_PIN_10|GPIO_PIN_11|GPIO_PIN_12, GPIO_PIN_SET);
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_SET);		
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_RESET);
 
	//根据ucLed的数值点亮相应的灯
	HAL_GPIO_WritePin(GPIOC, ucLed<<8, GPIO_PIN_RESET);
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_SET);		
	HAL_GPIO_WritePin(GPIOD, GPIO_PIN_2, GPIO_PIN_RESET);	
}

//按键扫描
uint8_t Key_Scan(void)
{
	uint8_t key_val=0;
	if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_0)==GPIO_PIN_RESET)
	{
		key_val=1;
	}
	if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_1)==GPIO_PIN_RESET)
	{
		key_val=2;
	}
	if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_2)==GPIO_PIN_RESET)
	{
		key_val=3;
	}
	if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==GPIO_PIN_RESET)
	{
		key_val=4;
	}
	return key_val;
}
/* USER CODE END 2 */

ADC

//adc.h
/* USER CODE BEGIN Prototypes */
uint16_t getADC2(void);
uint16_t getADC1(void);
/* USER CODE END Prototypes */

//adc.c
/* USER CODE BEGIN 1 */
uint16_t getADC1(void)
{
	uint16_t adc = 0;
	
	HAL_ADC_Start(&hadc1);
	adc = HAL_ADC_GetValue(&hadc1);
	
	return adc;
}
uint16_t getADC2(void)
{
	uint16_t adc = 0;
	
	HAL_ADC_Start(&hadc2);
	adc = HAL_ADC_GetValue(&hadc2);
	
	return adc;
}
/* USER CODE END 1 */

I2C

//i2c_hal.h
void iic_24c02_read(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum);
void iic_24c02_write(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum);

//i2c_hal.c
void iic_24c02_write(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum)
{
	I2CStart();
	I2CSendByte(0xa0);
	I2CWaitAck();
	
	I2CSendByte(ucAddr);
	I2CWaitAck();
	
	while(ucNum--)
	{
		I2CSendByte(*pucBuf++);
		I2CWaitAck();
	}
	I2CStop();
	HAL_Delay(500);
}

void iic_24c02_read(uint8_t* pucBuf, uint8_t ucAddr, uint8_t ucNum)
{
	I2CStart();
	I2CSendByte(0xa0);
	I2CWaitAck();
	I2CSendByte(ucAddr);
	I2CWaitAck();
	I2CStart();
	I2CSendByte(0xa1);
	I2CWaitAck();
	while(ucNum--)
	{
		*pucBuf++=I2CReceiveByte();
		if(ucNum)
			I2CSendAck();
		else
			I2CSendNotAck();
	}
	I2CStop();
}

其他

像 LCD、UART、TIM、RTC 就不用封装了,在 Main 中实现即可。

四、测试

烧录完,测试时,会发现设置时间时,闪烁的位置不对。

应该把 Lcd_Proc 的时间设置区的控制闪烁代码进行更改。

当然如果要居中显示,就要看看自己整了多少空格。

    // 根据当前设置项闪烁对应位置
    if (SETTING_TIME_Ctrl == 0x1)  // 闪烁状态
    {
        switch (Interface_Num)
        {
            case 0x10:  // 设置时(闪烁 HH 部分)
                Lcd_Disp_String[3] = ' ';  // 第1个数字
                Lcd_Disp_String[4] = ' ';  // 第2个数字
                break;
            case 0x11:  // 设置分(闪烁 MM 部分)
                Lcd_Disp_String[6] = ' ';  // 第1个数字
                Lcd_Disp_String[7] = ' ';  // 第2个数字
                break;
            case 0x12:  // 设置秒(闪烁 SS 部分)
                Lcd_Disp_String[9] = ' ';  // 第1个数字
                Lcd_Disp_String[10] = ' '; // 第2个数字
                break;
            default:
                break;
        }
    }

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

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

相关文章

爱普生FC-135晶振5G手机的极端温度性能守护者

在5G时代&#xff0c;智能手机不仅需要高速率与低延迟&#xff0c;更需在严寒、酷暑、振动等复杂环境中保持稳定运行。作为 5G 手机的核心时钟源&#xff0c;爱普生32.768kHz晶振FC-135凭借其宽温适应性、高精度稳定性与微型化设计&#xff0c;成为5G手机核心时钟源的理想选择&…

如何备份你的 Postman 所有 Collection?

团队合作需要、备份&#xff0c;还是迁移到其他平台&#xff0c;我们都需要在 Postman 中将这些珍贵的集合数据导出。 如何从 Postman 中导出所有集合(Collection)教程

MinGW下编译ffmpeg源码时生成compile_commands.json

在前面的博文MinGW下编译nginx源码中&#xff0c;有介绍到使用compiledb工具在MinGW环境中生成compile_commands.json&#xff0c;以为compiledb是捕获的make时的输出&#xff0c;而nginx生成时控制台是有输出编译时的命令行信息的&#xff0c;笔者之前编译过ffmpeg的源码&…

【数据结构】树与森林

目录 树的存储方法 双亲表示法 孩子表示法 孩子兄弟表示法 树、森林与二叉树的转换 树转换成二叉树 森林转换成二叉树 二叉树转换成森林 树与森林的遍历 树的遍历 森林的遍历 树的存储方法 双亲表示法 这种存储结构采用一组连续空间来存储每个结点&#xff0c;同时…

跟着StatQuest学知识08-RNN与LSTM

一、RNN &#xff08;一&#xff09;简介 整个过程权重和偏置共享。 &#xff08;二&#xff09;梯度爆炸问题 在这个例子中w2大于1&#xff0c;会出现梯度爆炸问题。 当我们循环的次数越来越多的时候&#xff0c;这个巨大的数字会进入某些梯度&#xff0c;步长就会大幅增加&…

【SpringCloud】Eureka的使用

3. Eureka 3.1 Eureka 介绍 Eureka主要分为两个部分&#xff1a; EurekaServer: 作为注册中心Server端&#xff0c;向微服务应用程序提供服务注册&#xff0c;发现&#xff0c;健康检查等能力。 EurekaClient: 服务提供者&#xff0c;服务启动时&#xff0c;会向 EurekaS…

初识MySQL · 数据类型

目录 前言&#xff1a; 数值类型 文本、二进制数据类型 时间类型 String类型 前言&#xff1a; 对于MySQL来说&#xff0c;是一门编程语言&#xff0c;可能定义不是那么的严格&#xff0c;但是对于MySQL来说也是拥有自己的数据类型的&#xff0c;比如tinyint&#xff0c;…

QT图片轮播器(QT实操学习2)

1.项目架构 1.UI界面 2.widget.h​ #ifndef WIDGET_H #define WIDGET_H#include <QWidget>#define TIMEOUT 1 * 1000 QT_BEGIN_NAMESPACE namespace Ui { class Widget; } QT_END_NAMESPACEclass Widget : public QWidget {Q_OBJECTpublic:Widget(QWidget *parent n…

深度解析衡石科技HENGSHI SENSE嵌入式分析能力:如何实现3天快速集成

嵌入式分析成为现代SaaS的核心竞争力 在当今SaaS市场竞争中&#xff0c;数据分析能力已成为产品差异化的关键因素。根据Bessemer Venture Partners的最新调研&#xff0c;拥有深度嵌入式分析功能的SaaS产品&#xff0c;其客户留存率比行业平均水平高出23%&#xff0c;ARR增长速…

杂草YOLO系列数据集4000张

一份开源数据集——杂草YOLO数据集&#xff0c;该数据集适用于农业智能化、植物识别等计算机视觉应用场景。 数据集详情 ​训练集&#xff1a;3,664张高清标注图像​测试集&#xff1a;180张多样性场景样本​验证集&#xff1a;359张严格筛选数据 下载链接 杂草YOLO数据集分…

Vue 2 探秘:visible 和 append-to-body 是谁的小秘密?

&#x1f680; Vue 2 探秘&#xff1a;visible 和 append-to-body 是谁的小秘密&#xff1f;&#x1f914; 父组件&#xff1a;identify-list.vue子组件&#xff1a;fake-clue-list.vue 嘿&#xff0c;各位前端探险家&#xff01;&#x1f44b; 今天我们要在 Vue 2 的代码丛林…

机器学习的一百个概念(1)单位归一化

前言 本文隶属于专栏《机器学习的一百个概念》&#xff0c;该专栏为笔者原创&#xff0c;引用请注明来源&#xff0c;不足和错误之处请在评论区帮忙指出&#xff0c;谢谢&#xff01; 本专栏目录结构和参考文献请见[《机器学习的一百个概念》 ima 知识库 知识库广场搜索&…

SpringCould微服务架构之Docker(5)

Docker的基本操作&#xff1a; 镜像相关命令&#xff1a; 1.镜像名称一般分两部分组成&#xff1a;[repository]:[tag]。 2. 在没有指定tag时&#xff0c;默认是latest&#xff0c;代表着最新版本的镜像。 镜像命令的案例&#xff1a; 镜像操作常用的命令&#xff1a; dock…

SpringAI与JBoltAI深度对比:从工具集到企业级AI开发范式的跃迁

一、Java生态下大模型开发的困境与需求 技术公司的能力断层 多数企业缺乏将Java与大模型结合的标准开发范式&#xff0c;停留在碎片化工具使用阶段。 大模型应用需要全生命周期管理能力&#xff0c;而不仅仅是API调用。 工具集的局限性 SpringAI作为工具集的定位&#xff1…

Python中multiprocessing的使用详解

1.实现多进程 代码实现&#xff1a; from multiprocessing import Process import datetime import timedef task01(name):current_timedatetime.datetime.now()start_timecurrent_time.strftime(%Y-%m-%d %H:%M:%S). "{:03d}".format(current_time.microsecond //…

强化学习与神经网络结合(以 DQN 展开)

目录 基于 PyTorch 实现简单 DQN double DQN dueling DQN Noisy DQN&#xff1a;通过噪声层实现探索&#xff0c;替代 ε- 贪心策略 Rainbow_DQN如何计算连续型的Actions 强化学习中&#xff0c;智能体&#xff08;Agent&#xff09;通过与环境交互学习最优策略。当状态空间或动…

飞书电子表格自建应用

背景 coze官方的插件不支持更多的飞书电子表格操作&#xff0c;因为需要自建应用 飞书创建文件夹 创建应用 开发者后台 - 飞书开放平台 添加机器人 添加权限 创建群 添加刚刚创建的机器人到群里 文件夹邀请群 创建好后&#xff0c;就可以拿到id和key 参考教程&#xff1a; 创…

深度学习四大核心架构:神经网络(NN)、卷积神经网络(CNN)、循环神经网络(RNN)与Transformer全概述

目录 &#x1f4c2; 深度学习四大核心架构 &#x1f330; 知识点概述 &#x1f9e0; 核心区别对比表 ⚡ 生活化案例理解 &#x1f511; 选型指南 &#x1f4c2; 深度学习四大核心架构 第一篇&#xff1a; 神经网络基础&#xff08;NN&#xff09; &#x1f330; 知识点概述…

MCP Server 实现一个 天气查询

​ Step1. 环境配置 安装 uv curl -LsSf https://astral.sh/uv/install.sh | shQuestion: 什么是 uv 呢和 conda 比有什么区别&#xff1f; Answer: 一个用 Rust 编写的超快速 (100x) Python 包管理器和环境管理工具&#xff0c;由 Astral 开发。定位为 pip 和 venv 的替代品…

Headless Chrome 优化:减少内存占用与提速技巧

在当今数据驱动的时代&#xff0c;爬虫技术在各行各业扮演着重要角色。传统的爬虫方法往往因为界面渲染和资源消耗过高而无法满足大规模数据采集的需求。本文将深度剖析 Headless Chrome 的优化方案&#xff0c;重点探讨如何利用代理 IP、Cookie 和 User-Agent 设置实现内存占用…