蓝桥杯单片机备战——第十一届省赛真题解析

2024-03-02 11:36

本文主要是介绍蓝桥杯单片机备战——第十一届省赛真题解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、第一部分

在这里插入图片描述在这里插入图片描述
注:第四题这种一般是没人记得清的,不过比赛的时候可以翻看STC15使用手册。如下:
在这里插入图片描述
在这里插入图片描述
注:在组合逻辑电路中,输出仅仅与输入有关;而在时序逻辑电路中,输出不仅与输入有关,还与之前的状态有关。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
注:以上属个人见解,有误的地方欢迎指出共同学习!

二、第二部分

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
上面题目用到的外设我们都封装过,还是很简单的,设计要求的时间就是用定时器前后台分配每个任务的运行频率嘛

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

简单分析一下题目,意思就是让我们显示一下采集到的温度数据,然后还能修改温度上下限,然后再根据上下限与实际温度的情况来输出DAC和点亮LED。 先了解了大体要求,是不是感觉也不是很难嘛,我们现在需要做的就是先实现大体框架,在逐一每个具体分支功能。好!我们现在就开始写代码!

1.数码管显示温度
还是老样子,我们先把代码框架构建好,然后再加入后续业务代码!
在这里插入图片描述

先初始化DS18B20,然后根据要求每0.5秒内获取并刷新一次温度数据。我们只需要用定时器前后台分配即可!
main.c

#include "main.h"bit KeyScan_Flag=0;
bit Refresh_Temp=0;
u8 temp=0;void main()
{System_Init();Timer1_Init();Init_DS18B20(); //DS18B20初始化while(1){if(KeyScan_Flag){ //50HZKey_Scan();KeyScan_Flag=0;}if(Refresh_Temp){ //4HZtemp = GetTemp();Refresh_Temp=0;}Show_Temp();}
}void Timer1_Isr(void) interrupt 3 //1ms中断一次
{static u8 count1=0,count2=0;if(++count1==20){	//20ms扫描一次按键KeyScan_Flag=1;count1=0;}if(++count2==250){Refresh_Temp=1;count2=0;}}

system.c

#include "system.h"void Timer1_Init(void)		//1毫秒@12.000MHz
{AUXR &= 0xBF;TMOD &= 0x0F;TL1 = 0x18;TH1 = 0xFC;TF1 = 0;ET1 = 1;EA = 1;TR1 = 1;
}void System_Init()//系统上电初始化
{//先锁存蜂鸣器,继电器所在573输出低电平,防止上电乱叫P25=1;P26=0;P27=1; //74HC138-->Y5=0,else=1-->Y5C=1,else=0P04=0;P06=0;	//ULN2003输入经过非门送入达林顿管,低电平有效P25=0;P26=0;P27=0;//锁存数据//关闭所有LED灯P25=0;P26=0;P27=1; //74HC138-->Y4=0,else=1-->Y4C=1,else=0P0=0XFF;P25=0;P26=0;P27=0;//锁存数据
}extern u8 temp;void Show_Temp()	//数据界面(显示温度)
{Nixie_Display(1,12); //提示符'C'Nixie_Display(7,temp%100/10); //显示温度十位Nixie_Display(8,temp%10);	//显示温度个位
}

现在就可以先编译一下然后烧录看看是否能正常显示温度,然后我们把温度设置界面也一起写了

u8 Tmax=30,Tmin=20;void Show_Range()	//显示参数设置界面
{Nixie_Display(1,17); //提示符'P'Nixie_Display(4,Tmax%100/10); //显示上限十位Nixie_Display(5,Tmax%10);	//显示上限个位Nixie_Display(7,Tmin%100/10); //显示下限十位Nixie_Display(8,Tmin%10);	//显示下限个位}

2.切换界面/设置参数
首先我们要确保按键已经切换到了独立按键模式,不然美分哦!然后看题目要求:
在这里插入图片描述
在这里插入图片描述

简单·,直接定义一个bit标志位,初始状态为0,按键按下翻转此标志位,此标志位为0则显示参数界面,为1则显示设置参数界面!
mian.c

#include "main.h"bit KeyScan_Flag=0;
bit Refresh_Temp=0;
bit Change_Flag=0;
u8 temp=0;void main()
{System_Init();Timer1_Init();Init_DS18B20(); //DS18B20初始化while(1){if(KeyScan_Flag){ //50HZKey_Scan();KeyScan_Flag=0;}if(Refresh_Temp){ //4HZtemp = GetTemp();Refresh_Temp=0;}if(Change_Flag) Show_Range();else Show_Temp();}
}void Timer1_Isr(void) interrupt 3 //1ms中断一次
{static u8 count1=0,count2=0;if(++count1==20){	//20ms扫描一次按键KeyScan_Flag=1;count1=0;}if(++count2==250){Refresh_Temp=1;count2=0;}}

key.c

#include "key.h"u8 Trg=0,Cont=0,Num=0;
bit LONG_CLICKED=0;/***************************************************
由于用到了长按判断,请务必让Key_Scan()/Keys_Scan()
函数每20ms运行一次,精度越高越好。如果不需要长按判断,
注释长按部分代码即可,此时不需要20ms一次
*************************************************///独立按键
void Key_Scan()	//20ms扫描一次
{//状态机u8 ReadData = P3^0XFF;Trg = ReadData&(ReadData^Cont);	//记录短按键值Cont = ReadData;	//记录长按键值if((Trg==0)&&(Cont!=0))	//记录按下时长{if(++Num==100) //满足长按2S{LONG_CLICKED=1;Num++;}}if((Trg==0)&&(Cont==0)) //松手或者未按下{//此处可扩展双击Num=0;}Key_Read();	//判断哪个按键按下
}extern bit Change_Flag;void Key_Read()
{if(Trg == 0x08){ //S4短按//按键任务区Change_Flag=!Change_Flag;}else if(Trg == 0x04){ //S5短按}else if(Trg == 0x02){	//S6短按}else if(Trg == 0x01){ //S7短按}else;if(!LONG_CLICKED) return;if(Cont==0x08){ //S4长按}else if(Cont==0x04){ //S5长按}else if(Cont==0x02){	//S6长按}else if(Cont==0x01){ //S7长按}else;LONG_CLICKED=0;
}

现在又可以烧录试试,看看能否正常切换且正常显示!然后我们再看题目要求:在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

同上面一样的,定义一个bit标志位,然后翻转就行,一个状态对应一个参数。然后修改上下限数值前先判断标志位来区分是改上限还是下限!

extern bit Change_Flag;	//切换显示内容标志位
bit Set_Flag=0;	//切换上下限标志位
extern u8 Tmax,Tmin;	//温度上下限
u8 Tmax_New=30,Tmin_New=20;void Key_Read()
{if(Trg == 0x08){ //S4短按//按键任务区Set_Flag=0;	//每次进入设置界面默认设置下限Change_Flag=!Change_Flag;if(!Change_Flag && Tmax_New>=Tmin_New)Tmax=Tmax_New,Tmin=Tmin_New;}else if(Trg == 0x04){ //S5短按Set_Flag =! Set_Flag;}else if(Trg == 0x02){	//S6短按if(Change_Flag){if(Set_Flag) Tmax_New++; //上限+1else Tmin_New++; //下限+1}}else if(Trg == 0x01){ //S7短按if(Change_Flag){if(Set_Flag) Tmax_New--; //上限+1else Tmin_New--; //下限+1}}else;if(!LONG_CLICKED) return;if(Cont==0x08){ //S4长按}else if(Cont==0x04){ //S5长按}else if(Cont==0x02){	//S6长按}else if(Cont==0x01){ //S7长按}else;LONG_CLICKED=0;
}

烧录代码试试效果,我们发现在设置的时候根本看不到上下限的变化,原因是我们在设置参数时,还是显示的Tmax,Tmin而不是Tmax_New,Tmin_New。所以在显示参数函数里面我们得加一个标志位来区分一下!

key.c

extern bit Change_Flag;	//切换显示内容标志位
bit Set_Flag=0;	//切换上下限标志位
bit ShowNew_Flag=0; //上下限显示区分标志位
extern u8 Tmax,Tmin;	//温度上下限
u8 Tmax_New=30,Tmin_New=20;void Key_Read()
{if(Trg == 0x08){ //S4短按//按键任务区Set_Flag=0;	//每次进入设置界面默认设置下限Change_Flag=!Change_Flag;if(!Change_Flag && Tmax_New>=Tmin_New){//上下限合理:Tmax=Tmax_New;Tmin=Tmin_New;}else if(!Change_Flag && Tmax_New<Tmin_New){ //不合理Tmax_New=Tmax;Tmin_New=Tmin;}ShowNew_Flag=0;	//显示真实上下限}else if(Trg == 0x04){ //S5短按Set_Flag =! Set_Flag;}else if(Trg == 0x02){	//S6短按if(Change_Flag){if(Set_Flag) Tmax_New++; //上限+1else Tmin_New++; //下限+1ShowNew_Flag=1;}}else if(Trg == 0x01){ //S7短按if(Change_Flag){if(Set_Flag) Tmax_New--; //上限-1else Tmin_New--; //下限-1ShowNew_Flag=1;}}else;if(!LONG_CLICKED) return;if(Cont==0x08){ //S4长按}else if(Cont==0x04){ //S5长按}else if(Cont==0x02){	//S6长按}else if(Cont==0x01){ //S7长按}else;LONG_CLICKED=0;
}

system.c

u8 Tmax=30,Tmin=20;
extern bit ShowNew_Flag;
extern u8 Tmax_New,Tmin_New;void Show_Range()	//显示参数设置界面
{if(!ShowNew_Flag){Nixie_Display(1,17); //提示符'P'Nixie_Display(4,Tmax%100/10); //显示上限十位Nixie_Display(5,Tmax%10);	//显示上限个位Nixie_Display(7,Tmin%100/10); //显示下限十位Nixie_Display(8,Tmin%10);	//显示下限个位}else{Nixie_Display(1,17); //提示符'P'Nixie_Display(4,Tmax_New%100/10); //显示上限十位Nixie_Display(5,Tmax_New%10);	//显示上限个位Nixie_Display(7,Tmin_New%100/10); //显示下限十位Nixie_Display(8,Tmin_New%10);	//显示下限个位}
}

烧录代码测试一下,🆗!完美实现功能!然后这里还有个小要求需要改一下:
在这里插入图片描述

	else if(Trg == 0x02){	//S6短按if(Change_Flag){if(Set_Flag) {if(++Tmax_New>99) Tmax_New=99; //上限+1}else {if(++Tmin_New>99) Tmin_New=99; //下限+1}ShowNew_Flag=1;}}else if(Trg == 0x01){ //S7短按if(Change_Flag){if(Set_Flag) {if(--Tmax_New<0) Tmax_New=0; //上限-1}else {if(--Tmin_New<0) Tmin_New=0; //下限-1}ShowNew_Flag=1;}}

至此,所有按键功能都调试好了!

3.DAC输出/点亮LED

我们再加入ADC与LED对应功能:


extern bit Set_error;void Set_DAC_LED()
{if(temp>Tmax){Write_DAC(205); //DAC输出4VSet_Leds(1,1);Set_Leds(2,0);Set_Leds(3,0); //点亮L1}else if(Tmin<=temp && temp<=Tmax){Write_DAC(154);	//DAC输出3VSet_Leds(1,0);Set_Leds(2,1);Set_Leds(3,0); //点亮L2}else if(temp<Tmin){Write_DAC(102);	//DAC输出2VSet_Leds(1,0);Set_Leds(2,0);Set_Leds(3,1); //点亮L3}else;if(Set_error) Set_Leds(4,1); //点亮L4else Set_Leds(4,0); //熄灭L4
}
if(Trg == 0x08){ //S4短按//按键任务区Set_Flag=0;	//每次进入设置界面默认设置下限Change_Flag=!Change_Flag;if(!Change_Flag && Tmax_New>=Tmin_New){//上下限合理:Tmax=Tmax_New;Tmin=Tmin_New;Set_error=0; //清零错误标志}else if(!Change_Flag && Tmax_New<Tmin_New){ //不合理Tmax_New=Tmax;Tmin_New=Tmin;Set_error=1; //置位错误标志}ShowNew_Flag=0;	//显示真实上下限}

现在我们烧录代码运行!应该是完美实现了题目所有要求,感觉不是很难,主要考察对于标志位的灵活运用!

如果有需要整个源工程代码的,欢迎评论区留言!

这篇关于蓝桥杯单片机备战——第十一届省赛真题解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/765983

相关文章

PostgreSQL的扩展dict_int应用案例解析

《PostgreSQL的扩展dict_int应用案例解析》dict_int扩展为PostgreSQL提供了专业的整数文本处理能力,特别适合需要精确处理数字内容的搜索场景,本文给大家介绍PostgreS... 目录PostgreSQL的扩展dict_int一、扩展概述二、核心功能三、安装与启用四、字典配置方法

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

使用Python绘制3D堆叠条形图全解析

《使用Python绘制3D堆叠条形图全解析》在数据可视化的工具箱里,3D图表总能带来眼前一亮的效果,本文就来和大家聊聊如何使用Python实现绘制3D堆叠条形图,感兴趣的小伙伴可以了解下... 目录为什么选择 3D 堆叠条形图代码实现:从数据到 3D 世界的搭建核心代码逐行解析细节优化应用场景:3D 堆叠图

深度解析Python装饰器常见用法与进阶技巧

《深度解析Python装饰器常见用法与进阶技巧》Python装饰器(Decorator)是提升代码可读性与复用性的强大工具,本文将深入解析Python装饰器的原理,常见用法,进阶技巧与最佳实践,希望可... 目录装饰器的基本原理函数装饰器的常见用法带参数的装饰器类装饰器与方法装饰器装饰器的嵌套与组合进阶技巧

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

全面解析MySQL索引长度限制问题与解决方案

《全面解析MySQL索引长度限制问题与解决方案》MySQL对索引长度设限是为了保持高效的数据检索性能,这个限制不是MySQL的缺陷,而是数据库设计中的权衡结果,下面我们就来看看如何解决这一问题吧... 目录引言:为什么会有索引键长度问题?一、问题根源深度解析mysql索引长度限制原理实际场景示例二、五大解决

深度解析Spring Boot拦截器Interceptor与过滤器Filter的区别与实战指南

《深度解析SpringBoot拦截器Interceptor与过滤器Filter的区别与实战指南》本文深度解析SpringBoot中拦截器与过滤器的区别,涵盖执行顺序、依赖关系、异常处理等核心差异,并... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现

深度解析Spring AOP @Aspect 原理、实战与最佳实践教程

《深度解析SpringAOP@Aspect原理、实战与最佳实践教程》文章系统讲解了SpringAOP核心概念、实现方式及原理,涵盖横切关注点分离、代理机制(JDK/CGLIB)、切入点类型、性能... 目录1. @ASPect 核心概念1.1 AOP 编程范式1.2 @Aspect 关键特性2. 完整代码实