来源:http://www.51hei.com/mcu/2477.html
什么是中断?就是打断当前要做的事,转而去执行别的事情。比如小七我现在正在电脑前写帖子,突然老妈叫我帮她下楼拿点东西,于是我就收到了老妈给我的一个中断(可以叫做外部中断),当我去拿东西时,突然尿急(内部中断,尿袋快要撑爆了),这又是一个中断,!我们把引起中断的事件叫做中断源(如老妈给我的任务,以及我的尿意。。。外部引起的叫外部中断,内部引起的叫内部中断),产生中断后就要去处理它,这称为中断的响应。
由于尿急这个内部中断的优先级比老妈给我的外部中断还要高,尿急了,我总得先去撒尿吧?所以我就先去执行撒尿这个语句(小七:怎么我觉得这个比喻很别扭呢?!)。当我撒完尿后(还是觉得很别扭。。。)我会返回来帮老妈拿东西(高优先级的中断处理完后返回执行优先级较低的中断),拿完东西了我再回到电脑前继续写帖子(全部中断处理完后继续接手中断前的工作)。这个就叫做中断的返回。这么通俗的比喻,大家对中断的概念应该都明白了吧,那么在单片机里面,中断有什么用呢?
当单片机正在执行程序的时候,突然某个按键按下了(产生外部中断),单片机就必须得去处理那个按键(中断的响应),看看是发生了什么事,按键处理完后继续回来执行程序(中断的返回)。
同样,单片机正在执行程序的时候,内部的定时器溢出(定时器后面会单独讲到),或者检测到单片机的电压低于正常值等等(单片机内部产生的中断叫内部中断),单片机就得去处理这些事情,然后再返回来。
在单片机里面,中断是有特殊的功能寄存器控制的,单片机里面一共有两个中断,一个是中断0,一个是中断1 ,和两个定时器T0,T1,定时器就是你打开它后,它会自动数数,当数到你给它限定的值时,它就会溢出,产生中断让CPU处理(就像一个桶,你打开水龙头后,水越来越多,当达到你需要的水位时,就会产生中断叫你去处理它)。这些我们先不深入了解他是什么东西,我们只需知道中断是用下面这几个关键词控制的就行了:
IT0 声明外部中断0的类型,IT0=1是边沿触发,0是电平触发
边沿触发就是当检测到外部电平发生变化,即由低变高,或者由高变低时,就会产生一个中断
电平触发就是检测到高电平或者低电平时,产生中断
IE0 外部边沿触发产生中断后,它的值会变1,当CPU响应后,会自动变为0
IT1 和IT0一样的含义
IE1 和IT0一样的含义
EX0 外部中断0控制器,EX0=1是允许外部中断,0是禁止外部中断,也就是不理会外部中断
ET0 这个是定时器中断控制器,ET1=1是允许定时器产生中断,0是禁止
EX1,ET1的含义跟上面的都一样。
EA 总中断控制器,1是允许有中断产生,0是禁止所有中断,就算天打雷劈也不理会
另外,还有一个中断优先级的控制器,就是控制是去帮妈妈拿东西的优先级高还是去撒尿的优先级高。
PX0 外部中断0的优先级控制,假如内外都产生了中断,1就是优先处理外部中断,0就是优先处理内部中断
PT0 定时器0优先级控制器,1就是优先相应定时器0
PT1 定时器1优先级控制器,1就是优先相应定时器1
另外还有串口的RI,TI,PS等我们先不学习了,不然大伙该乱了
(众人:其实我们早已凌乱了。。。一头雾水!)
还有个概念,就是中断请求的撤销,也就是说,产生中断后,会产生一个中断请求,为1,当CPU处理完中断后,必须清除这个请求,不然CPU又会认为这个中断没有处理又跑去处理它……
对于两个定时器产生的中断,当CPU响应后,会自动清除TF0,TF1这两个定制器中断请求,处理完后就跳出来,回到原来的地方继续执行。
对于外部中断INT0,INT1,如果中断类型是边沿触发,单片机会自动清除中断请求IE0,IE1
若是电平触发,如果有一个电平,使中断产生后,这个电平仍然还保持着,那么这个电平还会触发中断,这样CPU就死在中断的石榴裙下出不来了。。。
(众人:说了那么多,没例子你说个J8)
例子来啦!用外部中断来控制一个LED的亮灭。对了,外部中断并不是单片机的每个引脚都能产生,标有INT0或INT1的才行,我们看看11F02E的引脚图
中断的引脚是INT0:P3^2和 INT1:P3^3,我们用边沿触发(由高电平变成低电平时,就会触发)的中断方式来控制LED,
当我们没按下按键的时候,由于上拉电阻(不懂的问百度姐姐哦~)的原因,P3^2是高电平,当我们按下按键后,P3^2的电平就会变低,这个从高变低的过程就会产生一个中断(边沿触发),CPU会第一时间来相应这个中断,看看是谁看帖不回贴,看完帖子不评分,然后根据小七写的中断处理程序去处理他!
O(∩_∩)O 。
程序怎么写呢?
#include <reg52.h>
sbit led=P1^7; //定义LED
void zhongduan() interrupt 0 using 1 //声明中断处理函数,由于是外部中断,所以 interrupt X 里X的值是 0
{
led=!led; //CPU响应中断后会跑来这里执行(让led的状态取反)
}
void main() //主函数,程序执行的起点
{
EA=1; //允许CPU响应所有中断
IT0=1; //设外部中断0的响应模式为边沿触发
EX0=1; //允许中断0产生中断
while(1); //CPU不断在这里死循环,中断产生后放下工作去响应中断,处理完后然后再返回来继续死循环
}
按下按键,CPU会跑去中断处理函数执行,执行完中断处理后返回原处继续执行
(众人:这个中断跟我们前面学习的按键有什么区别么?)
当然有区别啦!虽然都是控制LED,但是按键是当CPU执行到按键检测如 if(key==0) 语句后,才去改变LED的状态,如果没有执行到,那么即使你按下按键单片机也不会响应的,也就是CPU主动去问按键有没有被按下。而中断呢,就是无论CPU在干嘛,只要触发中断后,CPU就会放下手中的活,第一时间赶回来处理,也就是按键被按下后主动告诉CPU。。。就像windows 系统的 ctrl+alt+del 组合键,你一按下这个组合键,无论系统在做什么,都会弹出任务管理器。
另外中断的处理函数是这样声明的
void abc() interrupt X using n
{
处理语句;
}
我们看到,只是普通的函数 加上了 interrupt X using Y 了而已,X 的取值是有规定的:
如果是外部中断0的中断处理函数,则X为0 即void abc() interrupt 0 using n
若是定时器0的中断处理函数,则 X 为1
若是外部中断1的中断处理函数,则 X 为2
若是定时器1的中断处理函数,则 X 为3
若是串口中断的中断处理函数,则 X 为4
n 是中断号,取值范围为 0 - 31
关于中断的学习,也到此告一段落了,当然还有一些问题没解决.......
Q1: 为什么count==40的时候数码管也不能闪烁???
/*实现目的:让LED灯以1000ms(即1s)产生流水灯效果,并用定时器0让数码管以500ms从0~F闪烁*/#include<reg52.h>#include<intrins.h>#define uint unsigned int#define uchar unsigned charsbit d1=P2^1;uchar weixuan=0x00;//位选全开uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//段选uchar temp,count,num;void delay(uint z){ uint x,y; for(x=z;x>0;x--){for(y=0;y<113;y++){}}}void main(){count=0;num=0;P1=weixuan;P0=table[num];temp=0xfe;P2=temp;TMOD=0x01;TH0=(65535-50000)/256;TL0=(65535-50000)%256;EA=1;ET0=1;TR0=1;while(1){delay(1000);temp=_crol_(temp,1);P2=temp;/*if(count==10){count=0;num++;if(num==16){num=0;}P0=table[num];} */}}void time0() interrupt 1{TH0=(65535-50000)/256;TL0=(65535-50000)%256;count++;if(count==10){count=0;num++;if(num==16){num=0;}P0=table[num];} }
/*PS:我们不能把数码管500ms闪烁时间是否到达的语句写在主程序中,若写在主程序中,有可能发生如下错误情况:当主程序在LED灯显示语句当中时,此时恰好定时器0进入中断并且count刚好加到了10,当定时器0中断再次进入时,主程序仍未退出LED流水灯的显示程序,那么此时count的值便变成了11,这样的话,count==10这个点永远检测不到,因此数码管闪烁失去了控制
在调试代码当中发现delay(uint z)函数与中断是同时执行的。。。 */