怎样建立网站快捷方式/微信公众号怎么推广

怎样建立网站快捷方式,微信公众号怎么推广,三个律师做网站合适吗,济南好的网站建设公司排名参考 第六届蓝桥杯嵌入式省赛程序设计题解析(基于HAL库)_蓝桥杯嵌入式第六届真题-CSDN博客 一、分析功能 RTC 定时 1)时间初始化 2)定时上报电压时间 ADC测量 采集电位器的输出电压信号。 串行功能 1)传送要设置…

参考

第六届蓝桥杯嵌入式省赛程序设计题解析(基于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");elsesprintf((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调整时间,其实按键这块只是实现时间到底后回到0if(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;}}elseucLed = 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();elseI2CSendNotAck();}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();elseI2CSendNotAck();}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.mzph.cn/web/73645.shtml

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

相关文章

第十二篇《火攻篇》:一把火背后的战争哲学与生存智慧

《孙子兵法》作为人类历史上最早的军事战略经典&#xff0c;其思想穿透了2500年的时空&#xff0c;至今仍在政治、商业乃至个人决策领域闪耀光芒。第十二篇《火攻篇》看似聚焦于具体的战术手段&#xff0c;实则蕴含了深刻的战争伦理与生存哲学。本文解读这一篇章如何用一把火点…

word光标一直闪的解决办法

在选项里&#xff0c;打开首选项&#xff0c;&#xff08;如果打不开&#xff0c;可以新建一个word也许就可以&#xff0c;实在不行只能靠眼疾手快&#xff0c;趁他还没闪赶紧点&#xff09; 选COM加载项&#xff0c;在里面取消勾选MicrosoftOfficePLUS

修改菜品-01.需求分析与设计

一.需求分析与设计 修改时要首先回显 设计时我们要设计哪些接口&#xff1f; 根据id查询菜品接口设计&#xff1a; 我们要根据id进行查询&#xff0c;因此在这里面id被作为路径参数。使用注解PathVariable。在查询菜品时&#xff0c;要将对应的口味也查出来&#xff0c;因此还…

动态IP与静态IP该如何选?

一、当IP地址成为"网络身份" 2023年亚马逊封号潮中&#xff0c;某杭州卖家因登录IP频繁切换&#xff08;早8点在纽约&#xff0c;午间瞬移到东京&#xff09;&#xff0c;触发平台风控导致账号冻结。这类"时空错乱症"揭示了跨境电商的生存法则&#xff1a…

【机器学习】——机器学习基础概念

摘要 本文主要介绍了机器学习的基础概念和典型过程。一个完整的机器学习过程包括数据收集、数据预处理、数据集划分、选择模型、训练模型、模型评估、模型优化和模型部署等关键步骤。在数据收集阶段&#xff0c;要获取足够且高质量的数据&#xff1b;数据预处理包括数据清理、…

麒麟信安全国产化智算一体机与海光C86芯片+ 海光DCU卡完成兼容性适配!

近日&#xff0c;麒麟信安全国产化智算一体机与国产海光C86芯片、海光DCU卡完成兼容性适配&#xff01; 在数字化转型的浪潮中&#xff0c;智能办公已成为企业提升效率、降低成本的重要手段&#xff0c;如何快速、高效地部署智能办公解决方案&#xff0c;成为许多企业面临的挑…

Axure设计之中继器表格——拖动列调整位置教程(中继器)

一、原理介绍 实现表格列的拖动排序&#xff0c;主要依赖Axure的动态面板和中继器两大核心功能&#xff1a; 动态面板交互控制 将表格的列标题封装在动态面板中&#xff0c;通过拖拽事件&#xff08;开始、移动、结束&#xff09;捕捉用户操作 在拖拽过程中实时计算鼠标位置&…

node-red dashboard

安装&#xff1a; npm install node-red-dashboard 访问&#xff1a; http://127.0.0.1:1880/ui 1. 创建一个新的 Dashboard 页面: 在 Node-RED 编辑器中&#xff0c;拖动一个 ui_dashboard 节点到工作区&#xff0c;并将其连接到你的数据流。 2. 配置 Dashboard 节点: 双击…

人体细粒度分割sapiens 实战笔记

目录 sapiens 分割示例: 分割config文件: 依赖项: mmcv安装 测试 cnn和ops一起测试: 报错: 保存图片代码: 人体box裁剪扩大,不裁剪扩大效果很差 sapiens https://github.com/facebookresearch/sapiens 分割示例: https://github.com/facebookresearch/sapie…

【cocos creator 3.x】3Dui创建,模型遮挡ui效果

官方文档&#xff1a;https://docs.cocos.com/creator/3.8/manual/zh/ui-system/components/editor/ui-model.html 1、3Dui创建 创建label&#xff0c;默认会添加canvas根节点和2dCamera 将Camera删除&#xff0c;canvas上组建去除cc.Canvas&#xff0c;cc.widget&#xff0…

车架号查询车牌号接口如何用Java对接

一、什么是车架号查询车牌号接口&#xff1f; 车架号查询车牌号接口&#xff0c;即传入车架号&#xff0c;返回车牌号、车型编码、初次登记日期信息。车架号又称车辆VIN码&#xff0c;车辆识别码。 二、如何用Java对接该接口&#xff1f; 下面我们以阿里云接口为例&#xff0…

SvelteKit 最新中文文档教程(12)—— 高级路由

前言 Svelte&#xff0c;一个语法简洁、入门容易&#xff0c;面向未来的前端框架。 从 Svelte 诞生之初&#xff0c;就备受开发者的喜爱&#xff0c;根据统计&#xff0c;从 2019 年到 2024 年&#xff0c;连续 6 年一直是开发者最感兴趣的前端框架 No.1&#xff1a; Svelte …

Ubuntu系统保姆级Paperless-ngx部署指南:零基础实现文档云端化管理

文章目录 前言1.关于Paperless-ngx2.Docker部署3.简单使用paperless4.安装cpolar内网穿透5. 配置公网地址6. 配置固定公网地址总结 前言 在当今快节奏的办公环境中&#xff0c;文档管理成为了一个不可忽视的问题。想象一下这样的场景&#xff1a;你需要一份重要的合同&#xf…

PostgREST实现DBaaS(数据库即服务)

目录 配置使用 验证 token使用 上文部署高可用PostgreSQL14集群后&#xff0c;本文介绍PostgREST&#xff0c;以及如何基于PostgREST实现数据库即服务&#xff0c;PostgREST可以在 PostgreSQL 数据库上通过解析数据库结构&#xff08;如表、视图、存储过程、权限等&#xff…

基于yolov11的铁路轨道铁轨缺陷检测系统python源码+pytorch模型+评估指标曲线+精美GUI界面

【算法介绍】 基于YOLOv11的铁路轨道铁轨缺陷检测系统是一种高效、准确的自动化检测技术&#xff0c;专门用于识别和检测铁轨上的各种缺陷。该系统利用YOLOv11这一先进的深度学习模型&#xff0c;实现了对Corrugation&#xff08;波纹磨耗&#xff09;、Spalling&#xff08;剥…

华为hcie证书考什么,怎么备考?

新盟教育 | 华为HALP授权培训合作伙伴 在ICT领域&#xff0c;华为HCIE证书是含金量极高的专业认证&#xff0c;它是对个人技术能力和专业素养的高度认可。对于渴望在网络、云计算、大数据等前沿领域深入发展的从业者而言&#xff0c;华为HCIE证书是一块强有力的职业敲门砖。 …

华为hcia——Datacom实验指南——配置IPv4静态路由,默认路由和浮动静态路由

什么是IPv4 IPv4静态路由&#xff0c;是手动配置的&#xff0c;不会随着网络拓扑的变化而变化&#xff0c;所配置的路由信息也不会在网络中传播&#xff0c;所以它主要运用在小型网络或者作为动态路由的补充。 IPv4的配置 配置的命令很简单 IP route-static &#xff08;目…

Linux一步部署主DNS服务器

​ #!/bin/bash #部署DHCP服务 #userli 20250319if [ "$USER" ! "root" ]then echo"错误&#xff1a;非root用户&#xff0c;权限不足&#xff01;"exit 0fi#防火墙与高级权限 systemctl stop firewalld && systemctl disable firewalld…

JumpServer:一款企业级开源堡垒机

在数字化运维时代&#xff0c;如何高效、安全地管控企业内部资产&#xff1f;JumpServer 作为一款完全开源的堡垒机&#xff0c;凭借其强大的 4A&#xff08;身份认证、授权控制、账号管理、安全审计&#xff09;能力与灵活的架构设计&#xff0c;实现事前授权、事中监察、事后…

STM32硬件IIC与OLED使用

OLED屏幕介绍 OLED即有机发光管(Organic Light-Emitting Diode,OLED)。OLED显示技术具有自发光、广视角、几乎无穷高的对比度、较低功耗、极高反应速度、可用于绕曲性面板、使用温度范围广、构造及制程简单等有点&#xff0c;被认为是下一代的平面显示屏新兴应用技术 OLED显示…