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

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

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机

OWASP十大安全漏洞解析

OWASP(开放式Web应用程序安全项目)发布的“十大安全漏洞”列表是Web应用程序安全领域的权威指南,它总结了Web应用程序中最常见、最危险的安全隐患。以下是对OWASP十大安全漏洞的详细解析: 1. 注入漏洞(Injection) 描述:攻击者通过在应用程序的输入数据中插入恶意代码,从而控制应用程序的行为。常见的注入类型包括SQL注入、OS命令注入、LDAP注入等。 影响:可能导致数据泄

从状态管理到性能优化:全面解析 Android Compose

文章目录 引言一、Android Compose基本概念1.1 什么是Android Compose?1.2 Compose的优势1.3 如何在项目中使用Compose 二、Compose中的状态管理2.1 状态管理的重要性2.2 Compose中的状态和数据流2.3 使用State和MutableState处理状态2.4 通过ViewModel进行状态管理 三、Compose中的列表和滚动

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

CSP 2023 提高级第一轮 CSP-S 2023初试题 完善程序第二题解析 未完

一、题目阅读 (最大值之和)给定整数序列 a0,⋯,an−1,求该序列所有非空连续子序列的最大值之和。上述参数满足 1≤n≤105 和 1≤ai≤108。 一个序列的非空连续子序列可以用两个下标 ll 和 rr(其中0≤l≤r<n0≤l≤r<n)表示,对应的序列为 al,al+1,⋯,ar​。两个非空连续子序列不同,当且仅当下标不同。 例如,当原序列为 [1,2,1,2] 时,要计算子序列 [

多线程解析报表

假如有这样一个需求,当我们需要解析一个Excel里多个sheet的数据时,可以考虑使用多线程,每个线程解析一个sheet里的数据,等到所有的sheet都解析完之后,程序需要提示解析完成。 Way1 join import java.time.LocalTime;public class Main {public static void main(String[] args) thro

ZooKeeper 中的 Curator 框架解析

Apache ZooKeeper 是一个为分布式应用提供一致性服务的软件。它提供了诸如配置管理、分布式同步、组服务等功能。在使用 ZooKeeper 时,Curator 是一个非常流行的客户端库,它简化了 ZooKeeper 的使用,提供了高级的抽象和丰富的工具。本文将详细介绍 Curator 框架,包括它的设计哲学、核心组件以及如何使用 Curator 来简化 ZooKeeper 的操作。 1

单片机毕业设计基于单片机的智能门禁系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍程序代码部分参考 设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订