> 技术文档 > C51 中断

C51 中断


1.外部中断

外部中断由单片机外部引脚的电平或边沿变化触发,适用于按键、传感器等外部事件响应。

1. 外部中断资源(8051 内核)

INT0:对应引脚 P3.2,中断号 0
INT1:对应引脚 P3.3,中断号 2
触发方式:电平触发(低电平有效)或边沿触发(下降沿 / 上升沿)

2. 外部中断配置步骤

步骤 1:配置触发方式(TCON 寄存器)
IT0(INT0 触发方式):IT0=0→电平触发;IT0=1→边沿触发(下降沿)
IT1(INT1 触发方式):IT1=0→电平触发;IT1=1→边沿触发(下降沿)

步骤 2:使能中断(IE 寄存器)
EX0=1(允许 INT0 中断);EX1=1(允许 INT1 中断)
EA=1(允许总中断,必须开启)

步骤 3:编写中断服务函数
格式:void 函数名() interrupt 中断号 { ... }
INT0 中断号:0;INT1 中断号:2

3. 外部中断实战:按键控制 LED 翻转

硬件连接:按键接 P3.2(INT0),LED 接 P2.0。

#include \"reg52.h\"typedef unsigned int u16;typedef unsigned char u8;sbit LED = P2^0;    // LED引脚sbit KEY_INT0 = P3^2;  // INT0按键引脚// 延时函数(用于消抖)void delay(u16 i) {    while(i--);}// INT0初始化函数void Int0Init() {    IT0 = 1;  // 边沿触发(下降沿)    EX0 = 1;  // 允许INT0中断    EA = 1;   // 允许总中断}// INT0中断服务函数void Int0Handler() interrupt 0 {    delay(1000);  // 消抖(约10ms)    if(KEY_INT0 == 0) {  // 确认按键按下        LED = ~LED;  // LED状态翻转    }}void main() {    Int0Init();  // 初始化中断    while(1) {        // 主程序可执行其他任务,中断自动响应    }}

2.定时器中断

定时器中断(以 T0 为例)
定时器中断由内部定时器计数溢出触发,适用于定时任务(如时钟、周期性采样)。

1. 定时器资源(8051 内核)

T0:定时器 0,中断号 1
T1:定时器 1,中断号 3
核心原理:通过计数内部机器周期,溢出时触发中断(可计算定时时间)

2. 定时器中断配置步骤

步骤 1:设置定时器模式(TMOD 寄存器)
选择工作模式(常用模式 1:16 位定时器,需手动重装初值
TMOD &= 0xF0; TMOD |= 0x01;(配置 T0 为模式 1)

步骤 2:计算并装载初值
机器周期 = 12 / 晶振频率(如 11.0592MHz 晶振,机器周期≈1.085μs)
定时时间 =(65536 - 初值)× 机器周期 → 初值 = 65536 - 定时时间 / 机器周期
装载初值到 TH0(高 8 位)和 TL0(低 8 位)

步骤 3:使能中断(IE 寄存器)
ET0=1(允许 T0 中断);ET1=1(允许 T1 中断)
EA=1(允许总中断)

步骤 4:启动定时器并编写中断服务函数
TR0=1(启动 T0);TR1=1(启动 T1)
中断服务函数:T0 中断号 1;T1 中断号 3

3. 定时器中断实战:LED 周期性闪烁

需求:LED 每 1 秒翻转一次(晶振 11.0592MHz)。
 

#include \"reg52.h\"typedef unsigned int u16;typedef unsigned char u8;sbit LED = P2^0;u16 cnt = 0;  // 计数变量(记录中断次数)// 定时器0初始化(定时10ms)void Timer0Init() {    TMOD &= 0xF0;  // 清除T0配置    TMOD |= 0x01;  // T0模式1(16位定时器)        // 初值计算:10ms = (65536 - 初值) × (12/11.0592MHz) → 初值≈55536    TH0 = 0xFC;  // 高8位:55536 / 256 = 217 → 0xFC    TL0 = 0x18;  // 低8位:55536 % 256 = 24 → 0x18        ET0 = 1;  // 允许T0中断    EA = 1;   // 允许总中断    TR0 = 1;  // 启动T0}// T0中断服务函数void Timer0Handler() interrupt 1 {    // 重新装载初值(模式1需手动重装)    TH0 = 0xFC;    TL0 = 0x18;        cnt++;  // 每10ms计数+1    if(cnt >= 100) {  // 10ms × 100 = 1000ms = 1s        cnt = 0;        LED = ~LED;  // 每秒翻转LED    }}void main() {    Timer0Init();  // 初始化定时器    while(1) {        // 主程序无需干预,定时器自动定时    }}