同学们在学习技能的时分,一定要多动脑筋,遇到问题后,三思而后问。有些时分你考虑的和真理就差一点点了,没有坚持下去,他人告知你后才茅塞顿开。这样得到的定论,能够让你学到常识,可是却培育不了你的逻辑思维能力。不是不能问,而是要在认真考虑的基础上再提问。
有同学有疑问,板子上只要 8 个流水灯,那假如我要做许多个流水灯一同把戏显现怎么办呢?那咱们在讲课的时分其实都说到过了,板子上是有 8 个流水灯,还有 6 个数码管,还有 1 个点阵 LED,一个数码管相当于 8 个小灯,一个点阵相当于 64 个小灯,那假如悉数算上的话,咱们板子上实践共接了 8+6*8+64=120 个小灯,你假如独自只接小灯,把戏灯就做出来了。
还有同学识,板子上流水灯和数码管能够一同作业吗?怎么一同作业呢?咱们刚说了,一个数码管是 8 个小灯,可是咱们反过来想一想,8 个流水灯不也便是相当于一个数码管吗。那板子上 6 个数码管咱们能够让他们一同亮,7 个数码管就不会了吗?当然了,考虑的习气是要渐渐培育的,想不到的同学持续尽力,每天行进一小步,坚持一段时刻后回头看看,就会发现你学会了许多。
咱们做了一个交通灯的程序给咱们做学习参阅。由于板子资源有限,所以我把左面 LED8和 LED9 一同亮作为绿灯,把中心 LED5 和 LED6 一同亮作为黄灯,把右边 LED2 和 LED3一同亮作为红灯,用数码管的低 2 位做倒计时,让 LED 和数码管一同参加作业。程序并不杂乱,也没有什么新常识点,咱们完全能够自己剖析了,然后下载编译试试看吧。
纯文本仿制
#include
sbit ADDR3 = P1^3;
sbit ENLED = P1^4;
unsigned char code LedChar[] = { //数码管显现字符转化表
0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
unsigned char LedBuff[7] = { //数码管+独立 LED 显现缓冲区
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};
bit flag1s = 1; //1 秒守时标志
unsigned char T0RH = 0; //T0 重载值的高字节
unsigned char T0RL = 0; //T0 重载值的低字节
void ConfigTimer0(unsigned int ms);
void TrafficLight();
void main(){
EA = 1; //开总中止
ENLED = 0; //使能数码管和 LED
ADDR3 = 1;
ConfigTimer0(1); //装备 T0 守时 1ms
while (1){
if (flag1s){ //每秒履行一次交通灯改写
flag1s = 0;
TrafficLight();
}
}
}
/* 装备并发动 T0,ms-T0 守时时刻 */
void ConfigTimer0(unsigned int ms){
unsigned long tmp; //暂时变量
tmp = 11059200 / 12; //守时器计数频率
tmp = (tmp * ms) / 1000; //核算所需的计数值
tmp = 65536 – tmp; //核算守时器重载值
tmp = tmp + 13; //补偿中止呼应延时形成的差错
T0RH = (unsigned char)(tmp>>8); //守时器重载值拆分为凹凸字节
T0RL = (unsigned char)tmp;
TMOD &= 0xF0; //清零 T0 的操控位
TMOD |= 0x01; //装备 T0 为形式 1
TH0 = T0RH; //加载 T0 重载值
TL0 = T0RL;
ET0 = 1; //使能 T0 中止
TR0 = 1; //发动 T0
}
/* 交通灯显现改写函数 */
void TrafficLight(){
static unsigned char color = 2; //色彩索引:0-绿色/1-黄色/2-赤色
static unsigned char timer = 0; //倒计时守时器
if (timer == 0){ //倒计时到 0 时,切换交通灯
switch (color){ //LED8/9 代表绿灯,LED5/6 代表黄灯,LED2/3 代表红灯
case 0: //切换到黄色,亮 3 秒
color = 1;
timer = 2;
LedBuff[6] = 0xE7;
break;
case 1: //切换到赤色,亮 30 秒
color = 2;
timer = 29;
LedBuff[6] = 0xFC;
break;
case 2: //切换到绿色,亮 40 秒
color = 0;
timer = 39;
LedBuff[6] = 0x3F;
break;
default:
break;
}
}else{ //倒计时未到 0 时,递减其计数值
timer–;
}
LedBuff[0] = LedChar[timer%10]; //倒计时数值个位显现
LedBuff[1] = LedChar[timer/10]; //倒计时数值十位显现
}
/* LED 动态扫描改写函数,需在守时中止中调用 */
void LedScan(){
static unsigned char i = 0; //动态扫描索引
P0 = 0xFF; //封闭一切段选位,显现消隐
P1 = (P1 & 0xF8) | i; //位选索引值赋值到 P1 口低 3 位
P0 = LedBuff[i]; //缓冲区中索引方位的数据送到 P0 口
if (i < 6){ //索引递加循环,遍历整个缓冲区
i++;
}else{
i = 0;
}
}
/* T0 中止服务函数,完结 LED 扫描和秒守时 */
void InterruptTimer0() interrupt 1{
static unsigned int tmr1s = 0; //1 秒守时器
TH0 = T0RH; //从头加载重载值
TL0 = T0RL;
LedScan(); //LED 扫描显现
tmr1s++; //1 秒守时的处理
if (tmr1s >= 1000){
tmr1s = 0;
flag1s = 1; //设置秒守时标志
}
}