了解51单片机中断系统、中断源、中断响应条件和优先级等,通过外部中断0实现按键控制LED亮灭为例理解中断工作原理和编程实现过程。
89C51/52的中断系统有5个中断源 ,2个优先级,可实现二级中断嵌套 。

INT0: 通过TCOM中的IT0来选择其为下降沿有效还是低电平有效。
T0: 定时计数溢出中断标志TF0置1
INT1: 通过TCOM中的IT1来选择其为下降沿有效还是低电平有效。
T1: 定时计数溢出中断标志TF1置1
串行口中断请求标志。当串行口接收完一帧串行数据时置位RI或当串行口发送完一帧串行数据时置位TI,向CPU申请中断。
| 位 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|---|---|---|---|---|---|---|---|---|
| 字节地址:88H | TF1 | TR1 | TF0 | TR0 | IE1 | IT1 | IE0 | IT0 |
CPU对中断系统所有中断以及某个中断源的开放和屏蔽是由中断允许寄存器IE控制的。
| 位 | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|---|---|---|---|---|---|---|---|---|
| 字节地址:A8H | EA | ES | ET1 | EX1 | ET0 | EX0 |
有时同一优先级中的中断申请不止一个,则有中断优先权排队问题。同一优先级的中断优先权排队,由中断系统硬件确定的自然优先级形成,其排列如所示

中断优先级有三条原则:
为了实现上述后两条原则,中断系统内部设有两个用户不能寻址的优先级状态触发器。其中一个置1,表示正在响应高优先级的中断,它将阻断后来所有的中断请求;另一个置1,表示正在响应低优先级中断,它将阻断后来所有的低优先级中断请求。

以上三条同时满足时,CPU才有可能响应中断。
你想使用的中断是哪个?选择相应的中断号;
你所希望的触发条件是什么?
你希望在中断之后干什么?
通过按键按下出发外部中断0来控制LED反转亮灭。

(对照这张图好理解)
使用外部中断0的代码:
默认按键为高电平,选择用下降沿触发中断
#include "reg52.h"
typedef unsigned int u16;
typedef unsigned char u8;
sbit led=P2^0;
sbit key3=P3^2;
//delay函数功能: 延时函数,i=1时,大约延时10us
void delay(u16 i)
{
while(i--);
}
//Int0Init函数功能:
void Int0Init()
{
IT0=1; //设置外部中断的触发方式下降沿触发
EX0=1; //开外部中断0允许位
EA=1; //打开总中断开关
}
//
void main()
{
Int0Init();
while(1);
}
//中断服务函数int0:
void int0 () interrupt 0
{
delay(1000);
if(key3==0)
{
led =~led;
}
}
不使用中断的代码:
//实现现象:下载程序后按下K1按键可以对D11小灯状态取反。
#include "reg52.h"//此文件中定义了单片机的特殊功能寄存器
typedef unsigned int u16;//对数据类型进行声明定义
typedef unsigned char u8;
sbit k1=P1^0; //定义P10口是k1
sbit led=P0^0; //定义P00口是led
//函数名 : delay
//函数功能: 延时函数,i=1时,大约延时10us,用定时器延时更精确
void delay(u16 i)
{
while(i--);
}
//函数名 : keypros
//函数功能: 按键处理函数,判断按键K1是否按下
void keypros()
{
if(k1==0) //检测按键K1是否按下
{
delay(1000); //消除抖动 一般大约10ms
if(k1==0) //再次判断按键是否按下
{
led=~led; //led状态取反
}
while(!k1); //检测按键是否松开
}
}
//主函数: main
void main()
{
led=0;
while(1)
{
keypros(); //按键处理函数
}
}