#include <reg51.h> unsigned char code table[]={0xFC,0x60,0xDA,0xF2,0x66,0xB6,0xBE,0xE0, 0XFE,0xf6,0xEE,0x3E,0x9C,0x7A,0x9E,0x8E}; unsigned char number = 0; int Count = 0; unsigned char i,j,flag = 0; sbit P1_2 = P1^2; sbit P1_3 = P1^3; sbit P1_4 = P1^4; void my_delay(unsigned int t) { while(t--); } void INT_0() interrupt 0 { P1_2 = 0; P1_3 = 0; P1_4 = 1; } void INT_1() interrupt 2 { P1_2 = 1; P1_3 = 0; P1_4 = 0; } void Service_Timer() interrupt 1 { TH0 = (65536 - 50000)/256; TL0 = (65536 - 50000)%256; number++; if(number == 20) { number = 0; Count--; } if(flag == 0) { P1_2 = 0; P1_3 = 0; P1_4 = 1; if(Count < 0) { Count = 10; flag++; } } if(flag == 1) { P1_2 = 1; P1_3 = 0; P1_4 = 0; if(Count < 0) { Count = 2; flag ++; } } if(flag == 2) { P1_2 = 0; P1_3 = 1; P1_4 = 0; if(Count < 0) { flag = 0; Count = 10; } } } void putch(unsigned char n) { SBUF=n; while(TI==0); TI=0; } void main(void) { TMOD = 0x01; TH0 = (65536 - 50000)/256; TL0 = (65536 - 50000)%256; ET0 = 1; EA = 1; TR0 = 1; IP = 0X04; EX0 = 1; IT0 = 0; EX1 = 1; IT1 = 0; while(1) { i=table[Count/10]; j=table[Count%10]; putch(i); putch(j); my_delay(200); } }代码详细解释
时间: 2024-04-25 20:27:02 浏览: 105
这段代码是一个基于8051单片机的计数器程序。程序中使用了两个外部中断,一个定时器中断和一个putch函数用于串口通信输出。程序通过三个LED灯来显示数字,每次定时器中断触发后数字会减少一。其中:
- code table[]数组用于存储数字0到9在七段数码管中的编码。
- INT_0()和INT_1()函数是外部中断0和1的中断服务函数,用于控制三个LED灯的亮灭。
- Service_Timer()函数是定时器中断的中断服务函数,用于触发计数器减少,并控制LED灯的亮灭。
- putch()函数用于将数字编码通过串口通信输出。
在主函数中,程序不断循环输出数字,并通过my_delay函数控制输出速度。
相关问题
#include <reg51.h> unsigned char code table[]={0xFC,0x60,0xDA,0xF2,0x66,0xB6,0xBE,0xE0,0xFE,0xF6,0xEE,0x3E,0x9C,0x7A,0x9E,0x8E}; unsigned char Count,i,j; void Timer0_Init() { TMOD |= 0x01; TH0 = 0x3C; TL0 = 0xAF; EA = 1; ET0 = 1; TR0 = 1; } void Timer0_ISR() interrupt 1 { TH0 = 0x3C; TL0 = 0xAF; Count--; if(Count == 60) { Count = 0; } i = table[Count / 10]; j = table[Count % 10]; SBUF = i; while(TI == 0); TI = 0; SBUF = j; while(TI == 0); TI = 0; } void main(void) { Count = 60; Timer0_Init(); while(1); }
这段程序与之前的程序相比,只有一个小错误。在定时器中断服务程序中,我们将`if(Count == 0)`修改为了`if(Count == 60)`,这样在计数器减到0时不会将计数器重新设置为60,而是一直保持在0。因此,需要将`if(Count == 60)`修改回`if(Count == 0)`。修改后的程序如下:
```c
#include <reg51.h>
unsigned char code table[]={0xFC,0x60,0xDA,0xF2,0x66,0xB6,0xBE,0xE0,0xFE,0xF6,0xEE,0x3E,0x9C,0x7A,0x9E,0x8E};
unsigned char Count,i,j;
void Timer0_Init()
{
TMOD |= 0x01;
TH0 = 0x3C;
TL0 = 0xAF;
EA = 1;
ET0 = 1;
TR0 = 1;
}
void Timer0_ISR() interrupt 1
{
TH0 = 0x3C;
TL0 = 0xAF;
Count--;
if(Count == 0)
{
Count = 60;
}
i = table[Count / 10];
j = table[Count % 10];
SBUF = i;
while(TI == 0);
TI = 0;
SBUF = j;
while(TI == 0);
TI = 0;
}
void main(void)
{
Count = 60;
Timer0_Init();
while(1);
}
```
在程序中,我们还是使用了定时器0来实现每隔1秒钟计数器减1的功能。程序中的`Timer0_Init()`函数用于初始化定时器0,`Timer0_ISR()`函数为定时器0的中断服务程序。在程序的`main()`函数中,我们调用了`Timer0_Init()`函数进行定时器0的初始化,并在无限循环中等待定时器0中断的触发。
#include <reg51.h> unsigned char code table[]={0xFC,0x60,0xDA,0xF2,0x66,0xB6,0xBE,0xE0, 0xFE,0xF6,0xEE,0x3E,0x9C,0x7A,0x9E,0x8E}; unsigned char Count,i,j; void Mydelay10ms(void) { unsigned char i,j; for(i=250;i>0;i--) for(j=248;j>0;j--); } void putch(unsigned char n) { SBUF=n; while(TI==0); TI=0; } void main(void) { Count=60; i=table[Count/10]; j=table[Count%10]; putch(i); putch(j); Mydelay10ms(); Mydelay10ms(); Mydelay10ms(); Mydelay10ms(); while(1) { Mydelay10ms(); Count--; if(Count==0) { Count=60; } i=table[Count/10]; j=table[Count%10]; putch(i); putch(j); Mydelay10ms(); Mydelay10ms(); Mydelay10ms(); Mydelay10ms(); } }上述程序修改为用定时中断,定时1s,实现0到60的计数,60清零
下面是修改后的程序:
```c
#include <reg51.h>
unsigned char code table[]={0xFC,0x60,0xDA,0xF2,0x66,0xB6,0xBE,0xE0,0xFE,0xF6,0xEE,0x3E,0x9C,0x7A,0x9E,0x8E};
unsigned char Count,i,j;
void Timer0_Init()
{
TMOD |= 0x01; // 设置定时器0为模式1
TH0 = 0x3C; // 设置定时器0初值为0x3C
TL0 = 0xAF; // 设置定时器0初值为0xAF
EA = 1; // 打开总中断开关
ET0 = 1; // 打开定时器0中断开关
TR0 = 1; // 启动定时器0
}
void Timer0_ISR() interrupt 1 // 定时器0中断服务程序
{
TH0 = 0x3C; // 重新设置定时器0初值
TL0 = 0xAF;
Count--; // 计数器减1
if(Count == 0)
{
Count = 60; // 计数器清零
}
i = table[Count / 10];
j = table[Count % 10];
SBUF = i; // 发送高位数码
while(TI == 0);
TI = 0;
SBUF = j; // 发送低位数码
while(TI == 0);
TI = 0;
}
void main(void)
{
Count = 60;
Timer0_Init();
while(1);
}
```
在修改后的程序中,我们使用了定时器0来实现每隔1秒钟计数器减1的功能。具体来说,我们设置定时器0为模式1,初值为0x3CAF,定时器溢出时触发定时器0中断服务程序,在其中让计数器减1并发送数码管显示的数码。程序中的`Timer0_Init()`函数用于初始化定时器0,`Timer0_ISR()`函数为定时器0的中断服务程序。程序中的`TR0`为定时器0的启动控制位,`ET0`为定时器0的中断允许控制位。在程序的`main()`函数中,我们调用了`Timer0_Init()`函数进行定时器0的初始化,并在无限循环中等待定时器0中断的触发。
阅读全文