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

文章讲述了如何使用STM32开发板配合DS18B20温度传感器实现温度数据采集、显示,以及通过按键控制温度上下限设置和LED灯状态。作者详细介绍了硬件初始化、定时器配置、数据处理和按键功能的编程过程。
摘要由CSDN通过智能技术生成

一、第一部分

在这里插入图片描述在这里插入图片描述
注:第四题这种一般是没人记得清的,不过比赛的时候可以翻看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){ //50HZ
			Key_Scan();
			KeyScan_Flag=0;
		}
		if(Refresh_Temp){ //4HZ
			temp = 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=0
	P04=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=0
	P0=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){ //50HZ
			Key_Scan();
			KeyScan_Flag=0;
		}
		if(Refresh_Temp){ //4HZ
			temp = 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++; //上限+1
			else Tmin_New++; //下限+1
		}
	}
	else if(Trg == 0x01){ //S7短按
		if(Change_Flag){
			if(Set_Flag) Tmax_New--; //上限+1
			else 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++; //上限+1
			else Tmin_New++; //下限+1
			ShowNew_Flag=1;
		}
	}
	else if(Trg == 0x01){ //S7短按
		if(Change_Flag){
			if(Set_Flag) Tmax_New--; //上限-1
			else Tmin_New--; //下限-1
			ShowNew_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输出4V
		Set_Leds(1,1);Set_Leds(2,0);Set_Leds(3,0); //点亮L1
	}
	else if(Tmin<=temp && temp<=Tmax){
		Write_DAC(154);	//DAC输出3V
		Set_Leds(1,0);Set_Leds(2,1);Set_Leds(3,0); //点亮L2
	}
	else if(temp<Tmin){
		Write_DAC(102);	//DAC输出2V
		Set_Leds(1,0);Set_Leds(2,0);Set_Leds(3,1); //点亮L3
	}
	else;
	
	if(Set_error) Set_Leds(4,1); //点亮L4
	else 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;	//显示真实上下限
	}

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

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

  • 21
    点赞
  • 35
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
由于蓝桥杯省赛的试题不允许公开,因此我们无法提供完整的试题及答案解析。以下是部分题目及答案解析供参考: 1. 两个数的和 题目描述: 输入两个整数,求它们的和。 输入格式: 共一行,包含两个整数。 输出格式: 共一行,包含一个整数,表示两个整数的和。 Java代码: import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); int a = in.nextInt(); int b = in.nextInt(); System.out.println(a + b); } } 2. 等差数列 题目描述: 输入一个整数n和一个整数d,输出1到n的等差数列,公差为d。 输入格式: 共一行,包含两个整数n和d。 输出格式: 共n行,每行一个整数,表示等差数列中的一个数。 Java代码: import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); int n = in.nextInt(); int d = in.nextInt(); int x = 1; for (int i = 0; i < n; i++) { System.out.println(x); x += d; } } } 3. 最长上升子序列 题目描述: 给定一个长度为n的整数序列,求它的最长上升子序列的长度。 输入格式: 共一行,包含n个整数,表示整数序列。 输出格式: 共一行,包含一个整数,表示最长上升子序列的长度。 Java代码: import java.util.Scanner; public class Main { public static void main(String[] args) { Scanner in = new Scanner(System.in); int n = in.nextInt(); int[] a = new int[n]; int[] f = new int[n]; for (int i = 0; i < n; i++) { a[i] = in.nextInt(); f[i] = 1; } int ans = 1; for (int i = 1; i < n; i++) { for (int j = 0; j < i; j++) { if (a[j] < a[i]) { f[i] = Math.max(f[i], f[j] + 1); } } ans = Math.max(ans, f[i]); } System.out.println(ans); } } 以上是部分题目及答案解析,仅供参考。建议考生在备战蓝桥杯省赛时,认真复习基础知识,多做题、多思考,提高解题能力。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值