分别利用查询和中断方式编程实现灯的闪烁控制实例

发布者:dswecd最新更新时间:2017-10-12 来源: eefocus关键字:查询  中断方式  闪烁控制 手机看文章 扫描二维码
随时随地手机看文章

本站前面有个灯的闪烁编程实例,那是用延时程序做的,现在回想起来,这样做不很恰当,为什么呢?我们的主程序做了灯的闪烁,就不能再干其它的事了,难道单片机只能这样工作吗?当然不是,我们能用定时器来实现灯的闪烁的功能。 

例1:查询方式 

ORG 0000H 

AJMP START 

ORG 30H 

START: 

MOV P1,#0FFH ;关所 灯 

MOV TMOD,#00000001B ;定时/计数器0工作于方式1 

MOV TH0,#15H 

MOV TL0,#0A0H ;即数5536 

SETB TR0 ;定时/计数器0开始运行 

LOOP:JBC TF0,NEXT ;如果TF0等于1,则清TF0并转NEXT处 

AJMP LOOP ;不然跳转到LOOP处运行 

NEXT:CPL P1.0 

MOV TH0,#15H 

MOV TL0,#9FH;重置定时/计数器的初值 

AJMP LOOP 

END AJMP LOOP 

END 

键入程序,看到了什么?灯在闪烁了,这可是用定时器做的,不再是主程序的循环了。简单地分析一下程序,为什么用JBC呢?TF0是定时/计数器0的溢出标记位,当定时器产生溢出后,该位由0变1,所以查询该位就可知宇时时间是否已到。该位为1后,要用软件将标记位清0,以便下一次定时是间到时该位由0变1,所以用了JBC指令,该指位在判1转移的同时,还将该位清0。 

以上程序是能实现灯的闪烁了,可是主程序除了让灯闪烁外,还是不能做其他的事啊!不,不对,我们能在LOOP:……和AJMP LOOP指令之间插入一些指令来做其他的事情,只要保证执行这些指令的时间少于定时时间就行了。那我们在用软件延时程序的时候不是也能用一些指令来替代DJNZ吗?是的,但是那就要求你精确计算所用指令的时间,然后再减去对应的DJNZ循环次数,很不方便,而现在只要求所用指令的时间少于定时时间就行,显然要求低了。当然,这样的办法还是不好,所以我们常用以下的办法来实现。 

程序2:用中断实现 

ORG 0000H , 

AJMP START 

ORG 000BH ;定时器0的中断向量地址


AJMP TIME0 ;跳转到真正的定时器程序处

ORG 30H


START:


MOV P1,#0FFH ;关所 灯


MOV TMOD,#00000001B ;定时/计数器0工作于方式1


MOV TH0,#15H 

MOV TL0,#0A0H ;即数5536 

SETB EA ;开总中断允许


SETB ET0 ;开定时/计数器0允许 

SETB TR0 ;定时/计数器0开始运行

LOOP: AJMP LOOP ;真正工作时,这里可写任意程序


TIME0: ;定时器0的中断处理程序

PUSH ACC

PUSH PSW ;将PSW和ACC推入堆栈保护


CPL P1.0


MOV TH0,#15H


MOV TL0,#0A0H ;重置定时常数


POP PSW

POP ACC

RETI


END


上面的例程中,定时时间一到,TF0由0变1,就会引发中断,CPU将自动转至000B处寻找程序并执行,由于留给定时器中断的空间只有8个字节,显然不足以写下所有有中断处理程序,所以在000B处安排一条跳转指令,转到实际处理中断的程序处,这样,中断程序能写在任意地方,也能写任意长度了。进入定时中断后,首先要保存当前的一些状态,程序中只演示了保存存ACC和PSW,实际工作中应该根据需要将可能会改变的单元的值都推入堆栈进行保护(本程序中实际不需保存护任何值,这里只作个演示)。 

上面的两个单片机程序运行后,我们发现灯的闪烁非常快,根本分辨不出来,只是视觉上感到灯有些晃动而已,为什么呢?我们能计算一下,定时器中预置的数是5536,所以每计60000个脉冲就是定时时间到,这60000个脉冲的时间是多少呢?我们的晶体震荡器是12M,所以就是60000微秒,即60毫秒,因此速度是非常快的。如果我想实现一个1S的定时,该怎么办呢?在该晶体震荡器濒率下,最长的定时也就是65。536个毫秒啊!上面给出一个例程。 

ORG 0000H 

AJMP START 

ORG 000BH ;定时器0的中断向量地址 

AJMP TIME0 ;跳转到真正的定时器程序处 

ORG 30H 
 
START: 

MOV P1,#0FFH ;关所 灯 

MOV 30H,#00H ;软件计数器预清0 

MOV TMOD,#00000001B ;定时/计数器0工作于方式1 

MOV TH0,#3CH 

MOV TL0,#0B0H ;即数15536 

SETB EA ;开总中断允许 

SETB ET0 ;开定时/计数器0允许 

SETB TR0 ;定时/计数器0开始运行 

LOOP: AJMP LOOP ;真正工作时,这里可写任意程序 

TIME0: ;定时器0的中断处理程序 

PUSH ACC 

PUSH PSW ;将PSW和ACC推入堆栈保护 

INC 30H 

MOV A,30H 

CJNE A,#20,T_RET ;30H单元中的值到了20了吗? 

T_L1: CPL P1.0 ;到了,取反P10 

MOV 30H,#0 ;清软件计数器 

T_RET: 

MOV TH0,#15H 

MOV TL0,#9FH ;重置定时常数 

POP PSW 

POP ACC 

RETI 

END 

先自己分析一下,看看是怎么实现的?这里采用了软件计数器的概念,思路是这样的,先用定时/计数器0做一个50毫秒的定时器,定时是间到了以后并不是立即取反P10,而是将软件计数器中的值加1,如果软件计数器计到了20,就取反P10,并清掉软件计数器中的值,不然直接返回,这样,就变成了20次定时中断才取反一次P10,因此定时时间就延长了成了20*50即1000毫秒了。 

这个思路在工程中是非常有用的,有的时候我们需要若干个定时器,可51中总共才有2个,怎么办呢?其实,只要这几个定时的时间有一定的公约数,我们就能用软件定时器加以实现,如我要实现P10口所接灯按1S每次,而P11口所接灯按2S每次闪烁,怎么实现呢?对了我们用两个计数器,一个在它计到20时,取反P10,并清零,就如上面所示,另一个计到40取反P11,然后清0,不就行了吗?这部份的程序如下

ORG 0000H


AJMP START

ORG 000BH ;定时器0的中断向量地址


AJMP TIME0 ;跳转到真正的定时器程序处

ORG 30H 

START:


MOV P1,#0FFH ;关所 灯 

MOV 30H,#00H ;软件计数器预清0 

MOV TMOD,#00000001B ;定时/计数器0工作于方式1


MOV TH0,#3CH


MOV TL0,#0B0H ;即数15536


SETB EA ;开总中断允许


SETB ET0 ;开定时/计数器0允许

SETB TR0 ;定时/计数器0开始运行


LOOP: AJMP LOOP ;真正工作时,这里可写任意程序


TIME0: ;定时器0的中断处理程序

PUSH ACC 

PUSH PSW ;将PSW和ACC推入堆栈保护


INC 30H


INC 31H ;两个计数器都加1

MOV A,30H


CJNE A,#20,T_NEXT ;30H单元中的值到了20了吗?

T_L1: CPL P1.0 ;到了,取反P10

MOV 30H,#0 ;清软件计数器 

T_NEXT:


MOV A,31H 

CJNE A,#40,T_RET ;31h单元中的值到40了吗? 

T_L2:


CPL P1.1

MOV 31H,#0 ;到了,取反P11,清计数器,返回


T_RET: 

MOV TH0,#15H

MOV TL0,#9FH ;重置定时常数


POP PSW


POP ACC m

RETI


END 


关键字:查询  中断方式  闪烁控制 引用地址:分别利用查询和中断方式编程实现灯的闪烁控制实例

上一篇:定时器/计数器的定时或计数范围
下一篇:51单片机计数显示程序举例

推荐阅读最新更新时间:2024-03-16 15:40

PIC16F877定时器0的中断方式调试
PIC16F877单片机定时器0的中断方式调试程序 #include pic.h #define uchar unsigned char #define uint unsigned int __CONFIG(0x3B31); void init(void); uint intnum; void main(void) { init(); while(1) { if(intnum = 20)//1S到时 { intnum = 0; RD0 = !RD0; } } } void init(void) { TRISD = 0; //PORTD引脚为输出 PORTD = 0; //POR
[单片机]
基于LPC2138控制的双向交通信号灯 (最后3s绿灯闪烁)
/**************************************************************************** * File:Main.C * 功能:计数器,通过两个按键来控制加减计数器,并输出数码管显示.。 ****************************************************************************/ #include config.h #include Math.h #define CON 0x00ffffff //控制P0.0~P0.23作为输出引脚 #define SW1 (1 30) #define SW2
[单片机]
基于LPC2138<font color='red'>控制</font>的双向交通信号灯 (最后3s绿灯<font color='red'>闪烁</font>)
单片机控制继电器 使LED灯交替闪烁
今天看到了继电器,想用89C51单片机控制它,使其能控制更丰富的电路,以完成一些复杂的功能。 首先用单片机的P2^0直接加到继电器上,发现没有反应,于是查了相关资料。发现单片机引脚输出的电流太小,不足以驱动继电器。 解决方法:利用三极管工作在饱和区的特性(开关特性)使继电器直接加在5伏电源之间,使其工作(或者说可以有足够的电流驱动它)。 注意:这里三极管并不是起放大电流的作用,只是当作一个开关。 原理图如图一(手画有点丑,见谅)。 连接电路前先不要接二极管D1,一会再解释二极管的作用。三极管是2n3904 NPN型三极管,基极电阻68欧姆,要把图中的5v电源vcc1和gnd1分别接到单片机的电源和地上., LED灯的电路由外
[单片机]
51学习之定时器中断的两种方式——查询中断
定时器中断可用两种方式处理:查询和进入中断处理。 1.查询 void Init_T0 (void) 定时器初始化 { TMOD = 0x01; TH0 = (65536-65000) /256; TL0 = (65536-65000) %6; EA = 1; ET0 = 1; TR0 = 1; } void Main() 主函数 { Init_T0 (); Speak1 =1; if(TF0==1) 查询中断标志 { Speak1=~Speak1; //TH0 = (65536-65000) /256; //TL0 = (65536-65000) %6; TF0=0; } } 2.进入中断处理 void Init_T0 (void
[单片机]
STM32 uC/OS_II 实践 之 任务调度过程理解及查询式事件
先把入口函数main给贴出来,就从这里开始,来自文件main.c /******************************************************************************* * Function Name : main * Description : 主函数,对系统以及硬件初始化,建立主函数并开启系统 * Input : None * Output : None * Return : None *******************************************************************************
[单片机]
ARM-Linux驱动--ADC驱动(中断方式
硬件平台:FL2440 内核版本:2.6.28 主机平台:Ubuntu 11.04 内核版本:2.6.39 原创作品,转载请标明出处:http://blog.csdn.net/yming0221/archive/2011/06/26/6568937.aspx 这个驱动写了好久,因为原来的Linux内核编译的时候将触摸屏驱动编译进内核了,而触摸屏驱动里的ADC中断在注册的时候类型选择的是 IRQF_SAMPLE_RANDOM,不是共享类型,所以,自己写的ADC驱动在每次open的时候,总提示ADC中断注册失败。 解决方案: 重新配置内核,选择触摸屏驱动以模块的形式编译,而不是直接编译进内核,这样Linux在启
[单片机]
ARM-Linux驱动--ADC驱动(<font color='red'>中断</font><font color='red'>方式</font>)
中断电平触发方式和跳变沿触发方式有什么区别
INT1的边沿触发只有下降沿有效,换句话说,只要出现了下降沿那肯定出现了低电平,所以对于单个中断来说,电平与边沿没区别。真正的区别在于:电平触发方式时,中断标志寄存器不锁存中断请求信号。也就是说,单片机把每个机器周期的S5P2采样到的外部中断源口线的电平逻辑直接赋值到中断标志寄存器。标志寄存器对于请求信号来说是透明的。这样当中断请求被阻塞而没有得到及时响应时,将被丢失。换句话说,要使电平触发的中断被CPU响应并执行,必须保证外部中断源口线的低电平维持到中断被执行为止。因此当CPU正在执行同级中断或更高级中断期间,产生的外部中断源(产生低电平)如果在该中断执行完毕之前撤销(变为高电平)了,那么将得不到响应,就如同没发生一样。同样,当C
[单片机]
PI推出可快速查询EuP Standards Finder
    Power Integrations公司宣布为电源设计师推出一个新的能效标准在线查询工具EuP Standards Finder。设计师可以利用该工具快速查询欧盟委员会的用能产品生态设计指令,再也不用为查找与自己相关的数据而进行耗时耗力的检索了。     EuP Standards Finder位于Power Integrations的网站上,用户可以先从A到Z的列表中选择一项应用,然后就可以立即看到待机、带载及空载功耗的Tier 1和Tier 2限值。搜索结果框内还可显示相关Power Integrations参考设计的链接。此外,用户还可阅读能效标准专家Mr. Green的评论,以便深入了解生态设计指令在具体
[电源管理]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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