微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > MCU和单片机设计讨论 > 单片机在这个程序的哪里进入中断

单片机在这个程序的哪里进入中断

时间:10-02 整理:3721RD 点击:
#include<reg52.h>
#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你看看是那条语句进入的

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top