第66节:单片机外部中断的基础

发布者:WhisperingHeart最新更新时间:2016-03-16 来源: eefocus关键字:单片机  外部中断 手机看文章 扫描二维码
随时随地手机看文章
开场白:

外部中断是单片机非常重要的内部资源,应用很广,它是单片机的高速开关感应器输入接口,它可以检测脉冲输入,可以接收红外遥控器的输入信号,可以检测高速运转的车轮或者电机圆周运动的反馈信号,可以检测输液器里瞬间即逝的水滴信号,可以接收模拟串口的数据信息,等等。

这一节要教大家两个知识点:

第一个:外部中断的初始化代码和中断函数的基本程序模板。

第二个:当系统存在两种中断以上时,如何设置外部中断0为最高优先级,实现中断嵌套功能。

具体内容,请看源代码讲解。

(1)硬件平台:

基于朱兆祺51单片机学习板。用S1按键作为模拟外部中断0的下降沿脉冲输入。原来S1按键是直接连接到P0^0口的,因此必须通过跳线把P0^0口连接到单片机外部中断0专用IO口P3^2上,只需把P0^0和P3^2的两根黄颜色跳冒去掉,通过一根线把P0^0和P3^2相互连接起来即可。这时每按下一次S1按键,就会给P3^2口产生一个下降沿的脉冲,然后程序会自动跳到中断函数中执行一次。

(2)实现功能:

用数码管低4位显示记录当前的下降沿脉冲数。用S1按键经过跳线后模拟外部中断0的下降沿输入,每按一次数码管就会显示往上累加的脉冲数。由于按键按下去的时候有抖动,也就按一次可能产生几个脉冲,所以按一次往往看到数据一次加了三四个,这种实验现象都是正常的。

(3)源代码讲解如下:

#include "REG52.H"

#define const_voice_short 40 //蜂鸣器短叫的持续时间

#define const_key_time1 20 //按键去抖动延时的时间

void initial_myself();

void initial_peripheral();

void delay_short(unsigned int uiDelayShort);

void delay_long(unsigned int uiDelaylong);

//驱动数码管的74HC595

void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01);

void display_drive(); //显示数码管字模的驱动函数

void display_service(); //显示的窗口菜单服务程序

//驱动LED的74HC595

void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01);

void T0_time(); //定时中断函数

void INT0_int();//外部0中断函数

sbit key_gnd_dr=P0^4; //模拟独立按键的地GND,因此必须一直输出低电平

sbit beep_dr=P2^7; //蜂鸣器的驱动IO口

sbit dig_hc595_sh_dr=P2^0; //数码管的74HC595程序

sbit dig_hc595_st_dr=P2^1;

sbit dig_hc595_ds_dr=P2^2;

unsigned char ucDigShow8; //第8位数码管要显示的内容

unsigned char ucDigShow7; //第7位数码管要显示的内容

unsigned char ucDigShow6; //第6位数码管要显示的内容

unsigned char ucDigShow5; //第5位数码管要显示的内容

unsigned char ucDigShow4; //第4位数码管要显示的内容

unsigned char ucDigShow3; //第3位数码管要显示的内容

unsigned char ucDigShow2; //第2位数码管要显示的内容

unsigned char ucDigShow1; //第1位数码管要显示的内容

unsigned char ucDigDot8; //数码管8的小数点是否显示的标志

unsigned char ucDigDot7; //数码管7的小数点是否显示的标志

unsigned char ucDigDot6; //数码管6的小数点是否显示的标志

unsigned char ucDigDot5; //数码管5的小数点是否显示的标志

unsigned char ucDigDot4; //数码管4的小数点是否显示的标志

unsigned char ucDigDot3; //数码管3的小数点是否显示的标志

unsigned char ucDigDot2; //数码管2的小数点是否显示的标志

unsigned char ucDigDot1; //数码管1的小数点是否显示的标志

unsigned char ucDigShowTemp=0; //临时中间变量

unsigned char ucDisplayDriveStep=1; //动态扫描数码管的步骤变量

unsigned char ucWd1Update=1; //窗口1更新显示标志

unsigned char ucWd=1; //本程序的核心变量,窗口显示变量。类似于一级菜单的变量。代表显示不同的窗口。本程序只有一个显示窗口

unsigned int uiPluseCnt=0; //本程序中累加中断脉冲数的变量

unsigned char ucTemp1=0; //中间过渡变量

unsigned char ucTemp2=0; //中间过渡变量

unsigned char ucTemp3=0; //中间过渡变量

unsigned char ucTemp4=0; //中间过渡变量

//根据原理图得出的共阴数码管字模表

code unsigned char dig_table[]=

{

0x3f, //0 序号0

0x06, //1 序号1

0x5b, //2 序号2

0x4f, //3 序号3

0x66, //4 序号4

0x6d, //5 序号5

0x7d, //6 序号6

0x07, //7 序号7

0x7f, //8 序号8

0x6f, //9 序号9

0x00, //无 序号10

0x40, //- 序号11

0x73, //P 序号12

};

void main()

{

initial_myself();

delay_long(100);

initial_peripheral();

while(1)

{

display_service(); //显示的窗口菜单服务程序

}

}

void display_service() //显示的窗口菜单服务程序

{

switch(ucWd) //本程序的核心变量,窗口显示变量。类似于一级菜单的变量。代表显示不同的窗口。

{

case 1: //显示第一个窗口的数据 本系统中只有一个显示窗口

if(ucWd1Update==1) //窗口1要全部更新显示

{

ucWd1Update=0; //及时清零标志,避免一直进来扫描

ucDigShow8=10; //第8位数码管显示无

ucDigShow7=10; //第7位数码管显示无

ucDigShow6=10; //第6位数码管显示无

ucDigShow5=10; //第5位数码管显示无

//先分解数据

ucTemp4=uiPluseCnt/1000;

ucTemp3=uiPluseCnt%1000/100;

ucTemp2=uiPluseCnt%100/10;

ucTemp1=uiPluseCnt%10;

//再过渡需要显示的数据到缓冲变量里,让过渡的时间越短越好

//以下增加的if判断就是略作修改,把整个4位数据中高位为0的去掉不显示。

if(uiPluseCnt<1000)

{

ucDigShow4=10; //如果小于1000,千位显示无

}

else

{

ucDigShow4=ucTemp4; //第4位数码管要显示的内容

}

if(uiPluseCnt<100)

{

ucDigShow3=10; //如果小于100,百位显示无

}

else

{

ucDigShow3=ucTemp3; //第3位数码管要显示的内容

}

if(uiPluseCnt<10)

{

ucDigShow2=10; //如果小于10,十位显示无

}

else

{

ucDigShow2=ucTemp2; //第2位数码管要显示的内容

}

ucDigShow1=ucTemp1; //第1位数码管要显示的内容

}

break;

}

}

void display_drive()

{

//以下程序,如果加一些数组和移位的元素,还可以压缩容量。但是鸿哥追求的不是容量,而是清晰的讲解思路

switch(ucDisplayDriveStep)

{

case 1: //显示第1位

ucDigShowTemp=dig_table[ucDigShow1];

if(ucDigDot1==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfe);

break;

case 2: //显示第2位

ucDigShowTemp=dig_table[ucDigShow2];

if(ucDigDot2==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfd);

break;

case 3: //显示第3位

ucDigShowTemp=dig_table[ucDigShow3];

if(ucDigDot3==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfb);

break;

case 4: //显示第4位

ucDigShowTemp=dig_table[ucDigShow4];

if(ucDigDot4==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xf7);

break;

case 5: //显示第5位

ucDigShowTemp=dig_table[ucDigShow5];

if(ucDigDot5==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xef);

break;

case 6: //显示第6位

ucDigShowTemp=dig_table[ucDigShow6];

if(ucDigDot6==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xdf);

break;

case 7: //显示第7位

ucDigShowTemp=dig_table[ucDigShow7];

if(ucDigDot7==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xbf);

break;

case 8: //显示第8位

ucDigShowTemp=dig_table[ucDigShow8];

if(ucDigDot8==1)

{

ucDigShowTemp=ucDigShowTemp|0x80; //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0x7f);

break;

}

ucDisplayDriveStep++;

if(ucDisplayDriveStep>8) //扫描完8个数码管后,重新从第一个开始扫描

{

ucDisplayDriveStep=1;

}

}

//数码管的74HC595驱动函数

void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01)

{

unsigned char i;

unsigned char ucTempData;

dig_hc595_sh_dr=0;

dig_hc595_st_dr=0;

ucTempData=ucDigStatusTemp16_09; //先送高8位

for(i=0;i<8;i++)

{

if(ucTempData>=0x80)dig_hc595_ds_dr=1;

else dig_hc595_ds_dr=0;

dig_hc595_sh_dr=0; //SH引脚的上升沿把数据送入寄存器

delay_short(1);

dig_hc595_sh_dr=1;

delay_short(1);

ucTempData=ucTempData<<1;

}

ucTempData=ucDigStatusTemp08_01; //再先送低8位

for(i=0;i<8;i++)

{

if(ucTempData>=0x80)dig_hc595_ds_dr=1;

else dig_hc595_ds_dr=0;

dig_hc595_sh_dr=0; //SH引脚的上升沿把数据送入寄存器

delay_short(1);

dig_hc595_sh_dr=1;

delay_short(1);

ucTempData=ucTempData<<1;

}

dig_hc595_st_dr=0; //ST引脚把两个寄存器的数据更新输出到74HC595的输出引脚上并且锁存起来

delay_short(1);

dig_hc595_st_dr=1;

delay_short(1);

dig_hc595_sh_dr=0; //拉低,抗干扰就增强

dig_hc595_st_dr=0;

dig_hc595_ds_dr=0;

}

void T0_time() interrupt 1 //定时器中断函数

{

TF0=0; //清除中断标志

TR0=0; //关中断

display_drive(); //数码管字模的驱动函数

TH0=0xfe; //重装初始值(65535-500)=65035=0xfe0b

TL0=0x0b;

TR0=1; //开中断

}

/* 注释一:

* 用朱兆祺51学习板中的S1按键作为模拟外部中断0的下降沿脉冲输入。

* 原来S1按键是直接连接到P0^0口的,因此必须通过跳线把P0^0口连接到

* 单片机外部中断0专用IO口P3^2上,只需把P0^0和P3^2的两个黄颜色跳冒去掉,通过一根

* 线把P0^0和P3^2相互连接起来即可。这时每按下一次S1按键,就会给P3^2口

* 产生一个下降沿的脉冲,然后程序会自动跳到以下中断函数中执行一次。

* 由于按键按下去的时候有抖动,也就按一次可能产生几个脉冲,所以按一次往往看到数据一次加了三四个,

* 这种实验现象都是正常的。

*/

void INT0_int(void) interrupt 0 //INT0外部中断函数

{

EX0=0; //禁止外部0中断 这个只是我个人的编程习惯,也可以不关闭

uiPluseCnt++; //累计外部中断下降沿的脉冲数

ucWd1Update=1; //窗口1更新显示

EX0=1; //打开外部0中断

}

void delay_short(unsigned int uiDelayShort)

{

unsigned int i;

for(i=0;i

{

; //一个分号相当于执行一条空语句

}

}

void delay_long(unsigned int uiDelayLong)

{

unsigned int i;

unsigned int j;

for(i=0;i

{

for(j=0;j<500;j++) //内嵌循环的空指令数量

{

; //一个分号相当于执行一条空语句

}

}

}

void initial_myself() //初始化单片机

{

/* 注释二:

* 矩阵键盘也可以做独立按键,前提是把某一根公共输出线输出低电平,

* 模拟独立按键的触发地,本程序中,把key_gnd_dr输出低电平。

* 朱兆祺51学习板的S1就是本程序中用到的一个独立按键。S1经过跳线后

* 连接到单片机的外部中断专用接口P3^2上,用来模拟外部下降沿脉冲输入。

*/

key_gnd_dr=0; //模拟独立按键的地GND,因此必须一直输出低电平

beep_dr=1; //用PNP三极管控制蜂鸣器,输出高电平时不叫。

TMOD=0x01; //设置定时器0为工作方式1

TH0=0xfe; //重装初始值(65535-500)=65035=0xfe0b

TL0=0x0b;

}

void initial_peripheral() //初始化外围

{

ucDigDot8=0; //小数点全部不显示

ucDigDot7=0;

ucDigDot6=0;

ucDigDot5=0;

ucDigDot4=0;

ucDigDot3=0;

ucDigDot2=0;

ucDigDot1=0;

EX0=1; //允许外部中断0

IT0=1; //下降沿触发外部中断0 如果是0代表低电平中断

/* 注释三:

* 注意,由于本系统中用了2个中断,一个是定时中断,一个是外部中断,

* 因此必须设置IP寄存器,让外部中断0为最高优先级,让外部中断0可以打断

* 定时中断。

*/

IP=0x01; //设置外部中断0为最高优先级,可以打断低优先级中断服务。实现中断嵌套功能

EA=1; //开总中断

ET0=1; //允许定时中断

TR0=1; //启动定时中断

}

总结陈词:

这节讲了外部中断的基本程序模板,下一节我会讲一个外部中断的实际应用项目例子。欲知详情,请听下回分解----利用外部中断实现模拟串口数据的收发。

关键字:单片机  外部中断 引用地址:第66节:单片机外部中断的基础

上一篇:第65节:大数据的除法运算
下一篇:第67节:利用外部中断实现模拟串口数据的收发

推荐阅读最新更新时间:2024-03-16 14:47

51单片机的LED与数码管的驱动
1、编写C51代码,实现流水灯字变花型: /*********************************************************** * 实验功能 : LED实现流水灯花型控制 *************************************************************/ # include reg51.h void delay(unsigned int z) { unsigned int x,y; for(x=z;x 0;x--) for(y=100;y 0;y--); } timer0() interrupt 1 { st
[单片机]
16-bit MCU实现超低功耗运动检测
谐振 LC 传感器技术用于运动检测已有数年,包括流量计量以及其它低速转动检测系统等。几乎在所有情况下,推动上述传感器设计发展的共同主线都是低功耗解决方案的需求,它通常为电池供电设备的低功耗解决方案。通过模拟测量组件与独立于主 CPU 工作的状态机处理接口相结合,本文以德州仪器 (TI) 的 MSP430FW42x 系列16位MCU为例,给出超低功耗运动检测系统解决方案的清晰说明。 图 1 显示了简易旋转运动检测系统的实施。除了微控制器与显示器之外,还显示了二通道谐振 LC 传感器的配置。单一传感器仅可用于转动检测,添加了第二个传感器后,就还可提供方向信息。 图 1 MSP430FW42x 转动系统原理图 传感器原理 使用
[测试测量]
16-bit <font color='red'>MCU</font>实现超低功耗运动检测
基于51单片机的智能语音电子秤设计
一.硬件方案 电子秤的测量原理是被称量物体的重量使传感器弹性体发生变形,输出与重量成正比的电信号,传感器输出信号经放大器放大后,输入转换器进行转换,转换成的频率信号直接送入微处理器中,其数字量由微机进行处理,而周边所需要的功能及各种接口电路也和微机连接应用,最后由显示屏幕以数字方式显示。 本设计硬件主要由51单片机+最小系统+LCD1602液晶显示模块+HX711模块+DS1302时钟电路+LED模块+语音模块+矩阵按键模块+10kg压力传感器及秤座。;如图: 二.设计功能 (1)用键盘设计单价; (2)称重后同时显示该物品的重量、单价和总额; (3)称重完之后有去皮功能; (4)能够实现总额的累加功能; (5)预存多种商
[单片机]
基于51<font color='red'>单片机</font>的智能语音电子秤设计
关于单片机堆栈概念的一个有趣的解释
因为单片机有CPU、存储器、IO等等,使他(人性化一点以配合下文)看起来就像一个比较小的计算机,所以,在理解单片机的时候如果能把你之前有的那些也许仅仅是直觉上的对计算机的理解融入进来的话,可能会对你学习单片机的概念有极大的帮助,至少对于我是这样的。 我想在关于单片机的众多让你头晕脑胀、摸不着头脑甚至想撞墙的概念里面, 堆栈 可能是其中最可恶的一个,因为即使单单是从汉语的角度来理解这个词就已经让你很晕了,其实我最初也想不通这是哪位大侠的创意,不过不用担心,这里我们完全不去讨论关于这个词的问题(这个词用得其实很好 堆 和 栈 都有他们各自的意思,准确的概括了这个区域的功能,有兴趣可以Baidu一下),这里我会打一个比较有趣的比方,以此来
[单片机]
STM32单片机/485通信详解
在了解485通信之前,需要先了解几个概念,以免一会儿晕头转向。我们在 【通信专栏】一:STM32串口通信(usart) 这篇文章中介绍了一点关于 通信分类 的内容: 串行通信与并行通信 串行通信 是指一比特一比特的收发数据,相对于 并行通信 可一次性收发N比特而言。所以串行或者并行通信都是一种概念,是理论层面的。 并行接口: 串行接口:串行接口必须的为GND,RXD,TXD三条线,其余几条作为握手用,可有可无。 串行接口 串口 ,全称 串行接口 ,也称串行通信接口(通常指COM接口),是 采用串行通信方式的扩展接口 ,乃是物理层面的,常见的有一般电脑应用的RS-232(使用 25 针或 9 针连接器)和工业电脑
[单片机]
STM32<font color='red'>单片机</font>/485通信详解
C51单片机学习笔记之矩阵键盘
简介 矩阵键盘一般为4×4或4×3的。矩阵键盘的判断方式分按行扫描和按列扫描。 简单说就是给全体一个高电平,然后给一个按键的一端附上低电平,再判断另一端是否为低电平。 原理图 代码部分 #include void delay()//延时函数 { unsigned int i; for(i=0;i 30000;i++); } unsigned char MatrixKey() { unsigned char KeyNumber;//定义一个返回值 P1=0xFF;//全体给高电平 P1_0=0;//一端给低电平 if(P1_4==0)//判断另一端是否为低电平 { delay(); while(P1
[单片机]
C51<font color='red'>单片机</font>学习笔记之矩阵键盘
提高单片机抗干扰:如何让你的设计少走弯路
  搞过产品的朋友都有体会,一个设计看似简单,硬件设计和代码编写很快就搞定,但在调试过程中却或多或少的意外,这些都是抗干扰能力不够的体现。   下面讨论一下如何让你的设计避免走弯路:   抗干扰体现在2个方面,一是硬件设计上,二是软件编写上。   这里重点提醒:在MCU设计中主要抗干扰设计是在硬件上,软件为辅。因为MCU的计算能力有限,所以要在硬件上花大工夫。   看看干扰的途径:   1:干扰信号干扰MCU的主要路径是通过I/O口,一是影响了MCU的数据采集,二是影响内部其它寄存器。   解决方法:后面讨论。   2:电源干扰:MCU虽然适应电压较宽(3-5。5V),但对于电源的波动却很敏感,比如说MCU可以在3V
[单片机]
MCU 面临 800V 电动汽车牵引逆变器的 3 种挑战
电动汽车 (EV) 牵引逆变器是电动汽车的核心。它将高压电池的直流电转换为多相(通常是三相)交流电来驱动牵引电机并控制制动能量的再生。电动汽车电子设备正在从 400V 架构转向 800V 架构,这正在逐步现实、普及,更高的电压会带来至少三个好处: - 快速充电 - 在相同电流下提供双倍的电量。 - 通过使用碳化硅 (SiC) 提高效率和功率密度。 - 通过使用更细的电缆来减轻重量,从而减少 800V 相同额定功率所需的电流。 在牵引逆变器中,微控制器(MCU)是系统的大脑,通过模数转换器(ADC)执行电机控制、电压和电流采样,使用磁芯计算磁场定向控制(FOC)算法,使用脉宽调制 (PWM) 信号驱动功率场效应晶体管
[嵌入式]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved