单片机在这个程序的哪里进入中断
#define uchar unsigned char
#define uint unsigned int
sbit key1=P3^4;
sbit key2=P3^5;
sbit key3=P3^6;
sbit key4=P3^7;
sbit dula=P2^6;
sbit wela=P2^7;
uchar code table[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
void delayms(uint);
uchar numt0,num;
void display(uchar numdis)
{
uchar shi,ge;
shi=numdis/10;
ge=numdis%10;
dula=1;
P0=table[shi];
dula=0;
P0=0xff;
wela=1;
P0=0xfe;
wela=0;
delayms(5);
dula=1;
P0=table[ge];
dula=0;
P0=0xff;
wela=1;
P0=0xfd;
wela=0;
delayms(5);
}
void delayms(uint xms)
{
uint i,j;
for(i=xms;i>0;i--)
for(j=110;j>0;j--);
}
void init()
{
TMOD=0x01;
TH0=(65536-45872)/256;
TL0=(65536-45872)%256;
EA=1;
ET0=1;
}
void keyscan()
{
if(key1==0)
{
delayms(10);
if(key1==0)
{
num++;
if(num==60)
num=0;
while(!key1);
}
}
if(key2==0)
{
delayms(10);
if(key2==0)
{
if(num==0)
num=60;
num--;
while(!key2);
}
}
if(key3==0)
{
delayms(10);
if(key3==0)
{
num=0;
while(!key3);
}
}
if(key4==0)
{
delayms(10);
if(key4==0)
{
while(!key4);
TR0=~TR0;
}
}
}
void main()
{
init();
while(1)
{
keyscan();
display(num);
}
}
void T0_time() interrupt 1
{
TH0=(65536-45872)/256;
TL0=(65536-45872)%256;
numt0++;
if(numt0==1)
{
numt0=0;
num++;
if(num==60)
num=0;
}
}
你的问题本身就是个问题,从你提出的问题来看,你还没有理解CPU响应中断的原理。记住:当某个中断源提出中断请求(中断请求是一个信号)时,硬件将置位相应的标志位,如果预先开放了CPU对这个中断源的响应,当CPU检测到标志位被置位后,CPU将暂停当前正在执行的程序代码,转而执行中断服务程序的代码。
你的程序中,中断源是定时器,当定时器计数溢出时会产生一个溢出脉冲(中断请求信号),该脉冲会将相应的中断标志位置位,CPU检测到标志位被置位后,将暂停当前正在执行的程序,转而去执行中断服务函数;当中断服务函数执行完成,CPU将回到暂停的程序开始继续执行。中断涉及到压堆栈和出堆栈。
现在回答你的问题:我不知道CPU从什么地方进入中断,因为我不知道定时器什么时候溢出,定时器溢出时CPU正在执行哪条语句。有一点我是知道的,一定是在你的main()函数中由while(1)构成的死循环中进入中断服务函数的。
你需要回去看中断系统一章的内容,并且你一定要知道为什么中断服务程序最后一条汇编指令是RETI。
void Timer0(void)
{
TMOD=0x01;//定时器0方式一,16位重装初值。
EA=1;
ET0=1;
TR0=1;
}
}
void Timer0_isr(void) interrupt 1
{
TH0=(65536-45872)/256;
TL0=(65536-45872)%256;
numt0++;
if(numt0==1)
{
numt0=0;
num++;
if(num==60)
num=0;
}
}
由于 TH0=(65536-45872)/256;
TL0=(65536-45872)%256;
当TH0,TL0 计数溢出时(即重装初值),这时产生中断,处理后面的程序;你这样书写程序好乱!不行的
#include<reg52.h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义
sbit KEY_ADD=P3^3; //定义按键输入端口
sbit KEY_DEC=P3^4;
#define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换
sbit LATCH1=P2^2;//定义锁存使能端口 段锁存
sbit LATCH2=P2^3;// 位锁存
unsigned char code dofly_DuanMa[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 显示段码值0~9
unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码
unsigned char TempData[8]; //存储显示值的全局变量
void DelayUs2x(unsigned char t);//函数声明
void DelayMs(unsigned char t);
void Display(unsigned char FirstBit,unsigned char Num);
void Init_Timer0(void);
/*------------------------------------------------
主函数
------------------------------------------------*/
void main (void)
{
unsigned char num=0,key_press_num;
KEY_ADD=1; //按键输入端口电平置高
KEY_DEC=1;
Init_Timer0();
while (1) //主循环
{
if(!KEY_ADD) //如果检测到低电平,说明按键按下
{
DelayMs(10); //延时去抖,一般10-20ms
if(!KEY_ADD) //再次确认按键是否按下,没有按下则退出
{
while(!KEY_ADD)
{
key_press_num++;
DelayMs(10); //10x200=2000ms=2s
if(key_press_num==200) //大约2s
{
key_press_num=0; //如果达到长按键标准
//则进入长按键动作
while(!KEY_ADD) //这里用于识别是否按
//键还在按下,如果按
//下执行相关动作,否则退出
{
if(num<999) //加操作
num++;
//即时把显示数据处理,如果去掉下面2
//句处理信息,实际上看不到渐变效果,
//而是看到跳变效果
//用户可以自行屏蔽测试
//分解显示信息,如要显示68,则68/10=6 68%10=8
TempData[0]=dofly_DuanMa[num/1000];
TempData[1]=dofly_DuanMa[num/100%10];
TempData[2]=dofly_DuanMa[num/10%10];
TempData[3]=dofly_DuanMa[num%10];
DelayMs(10);//用于调节长按循环操作
//的速度,可以自行调整此值以便达到最佳效果
}
}
}
key_press_num=0;//防止累加造成错误识别
if(num<999) //加操作
num++;
}
}
if(!KEY_DEC) //如果检测到低电平,说明按键按下
{
DelayMs(10); //延时去抖,一般10-20ms
if(!KEY_DEC) //再次确认按键是否按下,没有
//按下则退出
{
while(!KEY_DEC)
{
key_press_num++;
DelayMs(10);
if(key_press_num==200) //大约2s
{
key_press_num=0;
while(!KEY_DEC)
{
if(num>0) //减操作
num--;
//分解显示信息,如要显示68,则68/10=6 68%10=8
TempData[0]=dofly_DuanMa[num/1000];
TempData[1]=dofly_DuanMa[num/100%10];
TempData[2]=dofly_DuanMa[num/10%10];
TempData[3]=dofly_DuanMa[num%10];
DelayMs(20);
//用于调节长按循环操作的速度
}
}
}
key_press_num=0;//防止累加造成错误识别
if(num>0) //减操作
num--;
}
}
//分解显示信息,如要显示68,则68/10=6 68%10=8
TempData[0]=dofly_DuanMa[num/1000];
TempData[1]=dofly_DuanMa[num/100%10];
TempData[2]=dofly_DuanMa[num/10%10];
TempData[3]=dofly_DuanMa[num%10];
// Display(0,8); //显示全部8位
//主循环中添加其他需要一直工作的程序
}
}
void DelayUs2x(unsigned char t)
{
while(--t);
}
void DelayMs(unsigned char t)
{
while(t--)
{
//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);
}
}
void Display(unsigned char FirstBit,unsigned char Num)
{
static unsigned char i=0;
DataPort=0; //清空数据,防止有交替重影
LATCH1=1; //段锁存
LATCH1=0;
DataPort=dofly_WeiMa[i+FirstBit]; //取位码
LATCH2=1; //位锁存
LATCH2=0;
DataPort=TempData[i]; //取显示数据,段码
LATCH1=1; //段锁存
LATCH1=0;
i++;
if(i==Num)
i=0;
}
void Init_Timer0(void)
{
TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
//TH0=0x00; //给定初值
//TL0=0x00;
EA=1; //总中断打开
ET0=1; //定时器中断打开
TR0=1; //定时器开关打开
}
void Timer0_isr(void) interrupt 1
{
TH0=(65536-2000)/256; //重新赋值 2ms
TL0=(65536-2000)%256;
Display(0,8);
}
这是个类似的例子!供书写参考!
找到工程中的中断向量表,中断函数应该在那里被设置,不在这个文件中
每次50ms就会进入一次中断,如果晶振是11.0592的话每一条语句是1us你看看是那条语句进入的