一、第一部分
注:第四题这种一般是没人记得清的,不过比赛的时候可以翻看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; //显示真实上下限
}
现在我们烧录代码运行!应该是完美实现了题目所有要求,感觉不是很难,主要考察对于标志位的灵活运用!
如果有需要整个源工程代码的,欢迎评论区留言!