本套试题考察的难点还是在于按键逻辑的的编程设计上。
通过写出“设置键”和两个数码管的界面,就基本上可以打开本道题目的突破口了。
在数码管赋值函数里写上两个赋值界面,然后通过按键进行切换。
而矩阵按键看似很多,其实除去10和11两个按键有单独功能,剩下的都是取键值的简单逻辑。
main.c
定义一些函数变量,以及头文件
#include<stc15f2k60s2.h>
#include<ds18b20.h>
#define uchar unsigned char
#define uint unsigned int
/******定义函数*****/
uchar wendu ,wendu_max=30,wendu_min=20; //温度值 温度最大值 最小值
uchar wendu_flag;
uchar key_num=12; //按键键值
uchar tt=0 ,a=0; //中断计时器
uchar code tab[]={0xc0,0xf9,0xa4,0xb0,0x99,
0x92,0x82,0xf8,0x80,0x90,
0xbf,0xff};
uchar yi,er,san,si,wu,liu,qi,ba;
unsigned char jishu =0;//调温计数器
uchar wendu_huancun[4],shezhi_flag=0;
bit g;
板子初始化
/******板子初始化函数*******/
void lqb_init()
{
P2=((P2&0X1F)|0XA0);
P0=0X00;
P2&=0X1F;
P2=((P2&0X1F)|0X80);
P0=0XFF;
P2&=0X1F;
P2=((P2&0X1F)|0XC0);
P0=0XFF;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=0XFF;
P2&=0X1F;
}
延时函数
/********延时函数******/
void delay200um(uchar um)
{
uchar i,j,k;
for(i=0;i<um;i++)
for(j=0;j<3;j++)
for(k=0;k<35;k++);
}
继电器控制函数
/******继电器*****/
void jidianqi(bit kai)
{
if (kai)
{
P2=((P2&0X1F)|0XA0);P0=0X10;P2&=0X1F;
}
else if(!kai)
{
P2=((P2&0X1F)|0XA0);P0=0X00;P2&=0X1F;
}
P0=0XFF; //消隐
}
数码管显示程序
/******数码管一二位*****/
void display12(uchar yi,uchar er)
{
P2=((P2&0X1F)|0XC0);
P0=0X01;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[yi];
P2&=0X1F;
delay200um(5);
P2=((P2&0X1F)|0XC0);
P0=0X02;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[er];
P2&=0X1F;
delay200um(5);
}
/********数码管三四位*******/
void display34(uchar san,uchar si)
{
P2=((P2&0X1F)|0XC0);
P0=0X04;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[san];
P2&=0X1F;
delay200um(5);
P2=((P2&0X1F)|0XC0);
P0=0X08;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[si];
P2&=0X1F;
delay200um(5);
}
/******数码管五六位*******/
void display56(uchar wu,uchar liu)
{
P2=((P2&0X1F)|0XC0);
P0=0X10;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[wu];
P2&=0X1F;
delay200um(5);
P2=((P2&0X1F)|0XC0);
P0=0X20;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[liu];
P2&=0X1F;
delay200um(5);
}
/*********数码管七八位******/
void display78(uchar qi,uchar ba)
{
P2=((P2&0X1F)|0XC0);
P0=0X40;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[qi];
P2&=0X1F;
delay200um(5);
P2=((P2&0X1F)|0XC0);
P0=0X80;
P2&=0X1F;
P2=((P2&0X1F)|0XE0);
P0=tab[ba];
P2&=0X1F;
delay200um(5);
P0=0X00;
}
中断函数
/******中断函数 5毫秒****/
void Timer0Init(void) //5毫秒@11.0592MHz
{
AUXR |= 0x80; //定时器时钟1T模式
TMOD &= 0xF0; //设置定时器模式
TL0 = 0x00; //设置定时初值
TH0 = 0x28; //设置定时初值
TF0 = 0; //清除TF0标志
TR0 = 1; //定时器0开始计时
}
/*******中断处理函数*****/
void timer0() interrupt 1
{
tt++;
P0=0XFF;
if(tt==200) //0.2秒
{
tt=0;
a++;
g=~g;
if((wendu_flag==0)&&(a>=4)&&(g))// 0.8闪烁
{
a=0;
P2=((P2&0X1F)|0X80);
//P0=~(0x01); //亮
P00=0;
P2&=0X1F;
}
else if((wendu_flag==0)&&(a>=4)&&(!g))
{
a=0;
P2=((P2&0X1F)|0X80);
// P0=0xff; //灭
P00=1;
P2&=0X1F;
}
if((wendu_flag==1)&&(a==2)&&(g)) //0.4闪烁
{
a=0;
P2=((P2&0X1F)|0X80);
// P0=~(0x01); //亮
P00=0;
P2&=0X1F;
}
else if (((wendu_flag==1)&&(a==2)&&(!g)))
{
a=0;
P2=((P2&0X1F)|0X80);
//P00=0xff; //灭
P00=1;
P2&=0X1F;
}
if((wendu_flag==1)&&(a==1)&&(g)) //0.2闪烁
{
a=0;
P2=((P2&0X1F)|0X80);
// P0=~(0x01); //亮
P00=0;
P2&=0X1F;
}
else if((wendu_flag==1)&&(a==1)&&(!g))
{
a=0;
P2=((P2&0X1F)|0X80);
// P0=0xff; //灭
P00=0;
P2&=0X1F;
}
}
}
按键捕获函数
/******key16******/
void key_scan12()
{
uchar temp;
P44=0;P42=1;P3=0X7f;temp=P3;temp&=0x0f;
if(temp!=0x0f)
{
delay200um(20);
P44=0;P42=1;P3=0X7f;temp=P3;temp&=0x0f;
if(temp!=0x0f)
{
switch (temp)
{
case 0x0e:key_num=0; //0
break;
case 0x0d :key_num=3; //3
break;
case 0x0b :key_num=6; //6
break;
case 0x07 :key_num=9; //9
break;
default:break;
}
while(P3!=0x0f)
{
temp=P3;
temp=temp&0x0f;
}
}
}
P44=1;P42=0;P3=0Xbf;temp=P3;temp&=0x0f;
if(temp!=0x0f)
{
delay200um(20);
P44=1;P42=0;P3=0Xbf;temp=P3;temp&=0x0f;
if(temp!=0x0f)
{
switch (temp)
{
case 0x0e:key_num=1; //1
break;
case 0x0d:key_num=4; //4
break;
case 0x0b:key_num=7; //7
break;
case 0x07:key_num=10; //设置
break;
default:break;
}
}
while(P3!=0x0f)
{
temp=P3;temp=temp&0x0f;
}
}
P44=1;P42=1;P3=0Xdf;temp=P3;temp&=0x0f;
if(temp!=0x0f)
{
delay200um(20);
P44=1;P42=1;P3=0Xdf;temp=P3;temp&=0x0f;
if(temp!=0x0f)
{
switch (temp)
{
case 0x0e:key_num=2; //2
break;
case 0x0d:key_num=5; //5
break;
case 0x0b:key_num=8; //8
break;
case 0x07:key_num=11; //清除
break;
default:break;
}
}
while(P3!=0x0f)
{
temp=P3;temp=temp&0x0f;
}
}
}
主函数
/******主函数****/
void main()
{
lqb_init();
Timer0Init(); //中断
while(1)
{
EA=1;
ET0=1;
wendu=ds18b20_init();
yi=10;er=10;san=10;si=10;wu=10;liu=10;
qi=wendu/10;ba=wendu%10;
key_scan12();
//温度模式检测
if(wendu<wendu_min)
{
wendu_flag=0;
yi=10;er=0; san=10;si=11;wu=11;liu=11; //温度区间0
jidianqi(0); //继电器关
}
else if(wendu>wendu_max)
{
wendu_flag=2;
yi=10;er=2;san=10;si=11;wu=11;liu=11; //温度区间2
jidianqi(1); //继电器开
}
else if((wendu>wendu_min)&&(wendu<wendu_max))
{
wendu_flag=1;
yi=10;er=1;san=10;si=11;wu=11;liu=11; //温度区间1;
jidianqi(0); //继电器关
}
if (key_num==10) //按键按下设置后进入设置模式
{
shezhi_flag=1;
while(1)
{
key_scan12();
if(jishu<4)
{
wendu_huancun[jishu]=key_num;
jishu++; // 没进入一次计数加一
if(key_num==11)
{
jishu=0; //清除键 计数清零
}
}
if((shezhi_flag=1)&&(key_num==10))
{
wendu_max=(wendu_huancun[0]*10)+wendu_huancun[1];
wendu_min=(wendu_huancun[2]*10)+wendu_huancun[3];
jishu=0;
}
if(wendu_max<=wendu_min) //最大温度如果小于最小温度 L2亮
{
P2=((P2&0X1F)|0X80);
P02=0;
P2&=0X1F;
}
else
{
P2=((P2&0X1F)|0X80);
P02=1;
P2&=0X1F;
}
yi=10;er=wendu_max/10;san=wendu_max%10;si=11;
wu=11;liu=10;qi=wendu_min/10;ba=wendu_min%10;
display12(yi,er);
display34(san,si);
display56(wu,liu);
display78(qi,ba);
}
}
display12(yi,er);
display34(san,si);
display56(wu,liu);
display78(qi,ba);
}
}
ds18b20.c
这一部分 官方给的有驱动,我们自己只用写一个初始化函数把温度值读出来就行了,唯一要注意的是 15单片机运行速度是51的12倍,我们要修改一下延时函数。
unsigned char ds18b20_init()
{
unsigned char temp ,low,gao;
init_ds18b20();
Write_DS18B20(0xcc);
Write_DS18B20(0x44);
Delay_OneWire(200);
init_ds18b20();
Write_DS18B20(0xcc);
Write_DS18B20(0xbe);
low=Read_DS18B20(); //高位
gao=Read_DS18B20(); //低位
temp=((gao<<4)|(low>>4));
return temp;
}