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

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

相关文章

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

Python 中的异步与同步深度解析(实践记录)

《Python中的异步与同步深度解析(实践记录)》在Python编程世界里,异步和同步的概念是理解程序执行流程和性能优化的关键,这篇文章将带你深入了解它们的差异,以及阻塞和非阻塞的特性,同时通过实际... 目录python中的异步与同步:深度解析与实践异步与同步的定义异步同步阻塞与非阻塞的概念阻塞非阻塞同步

Redis中高并发读写性能的深度解析与优化

《Redis中高并发读写性能的深度解析与优化》Redis作为一款高性能的内存数据库,广泛应用于缓存、消息队列、实时统计等场景,本文将深入探讨Redis的读写并发能力,感兴趣的小伙伴可以了解下... 目录引言一、Redis 并发能力概述1.1 Redis 的读写性能1.2 影响 Redis 并发能力的因素二、

Spring MVC使用视图解析的问题解读

《SpringMVC使用视图解析的问题解读》:本文主要介绍SpringMVC使用视图解析的问题解读,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring MVC使用视图解析1. 会使用视图解析的情况2. 不会使用视图解析的情况总结Spring MVC使用视图

利用Python和C++解析gltf文件的示例详解

《利用Python和C++解析gltf文件的示例详解》gltf,全称是GLTransmissionFormat,是一种开放的3D文件格式,Python和C++是两个非常强大的工具,下面我们就来看看如何... 目录什么是gltf文件选择语言的原因安装必要的库解析gltf文件的步骤1. 读取gltf文件2. 提

Java中的runnable 和 callable 区别解析

《Java中的runnable和callable区别解析》Runnable接口用于定义不需要返回结果的任务,而Callable接口可以返回结果并抛出异常,通常与Future结合使用,Runnab... 目录1. Runnable接口1.1 Runnable的定义1.2 Runnable的特点1.3 使用Ru

使用EasyExcel实现简单的Excel表格解析操作

《使用EasyExcel实现简单的Excel表格解析操作》:本文主要介绍如何使用EasyExcel完成简单的表格解析操作,同时实现了大量数据情况下数据的分次批量入库,并记录每条数据入库的状态,感兴... 目录前言固定模板及表数据格式的解析实现Excel模板内容对应的实体类实现AnalysisEventLis

Java的volatile和sychronized底层实现原理解析

《Java的volatile和sychronized底层实现原理解析》文章详细介绍了Java中的synchronized和volatile关键字的底层实现原理,包括字节码层面、JVM层面的实现细节,以... 目录1. 概览2. Synchronized2.1 字节码层面2.2 JVM层面2.2.1 ente