1、 本文是对与以前发表的程序做个适当更改,因为在使用的过程中出现了一个问题,也就是在外部中断的处理函数里面处理欠妥。在执行外部中断函数的时候,本意以定时器中断来代表解码超时,可实际情况是,即使解码超时,定时器产生溢出,但是程序却不能按照正常程序跳进定时器中断,执行定时中断函数,本以为是中断优先级的原因,但是在设置了定时器中断优先级为最高,也还是不能正常进入。随后本人想到了解决办法,利用定时器 0 溢出标志 TF0 作为判断超时的依据,因为当 TF0 为 1 时,定时器向 CUP 申请中断,我们可以通过查询的方式,当 TF0 为 1,则解码超时,退出解码。此时不需要定时器中断函数,也可判断解码是否
2、超时。有人就说了,那我判断超时是否我可以定义变量,当变量超过一定数值时代表超时。我想说,何必呢,本身定时器 0 溢出标志就是 TF0,红外解码用到定时器,那不就刚好么,何必再去定义变量呢。按照了解,通用的红外信号电平持续最长的时间是起始电平,一个 9ms 的低电平时间,定时器从 0 开始计时到溢出,如果是 52单片机 12M 晶振速度,按照 12 分频,一个定时器脉冲时间是 1us,溢出所需要的时间是65ms 左右。所以当定时器溢出,我们就不必等待,TF0 作为停止解码的标志也只能用在这种速度比较低的单片机上,对于高速单片机,假如定时器溢出只需要 2ms,那要是再使用TF0 做判断,也就无意义
3、了。好了,废话不多说了。注:主函数由读者自己编写,本程序只用于解码 ,最终得出的按键码存于变量Key_Temp 中。 本人习惯写某个器件的程序的时候,习惯创建.c 和.h 文件,因为这样方便移植,在新建的程序中只需添加.C 文件和 include .h 文件就行了,很方便。 本人用芯片是 STC15W4K32S4。自己用这块芯片做有开发板,平时写程序都用它。也可用一般的 51 单片机,但要注意定时器的时间和外部中断的端口。本程序只适用 NEC 编码的解码,RC5 编码请绕道。以下是头文件信息:#ifndef _IR_H_#define _IR_H_sbit IR = P36;extern un
4、signed char Long_Press,Long_Press1; /长按计数,用作长按判断extern unsigned char Key_Temp; /保存按键变量void Init_IR(); /初始化红外有关参数#endif以下是.c 文件:#include /也可以用 reg52.h ,一般的编译器没有这个头文件,因为这是我个人添加进编译器里面的 #include “IR.h“ #include “IR.h“/*时间计算:时间由定时器计的定时值来决定,本程序设定工作频率是 24MHz,但定时器是 12T 模式。故一个定时脉冲时间为 0.5us。实际定时所得时间应为(TH08000
5、) /先右移一位TH0 = 0;TL0 = 0;TR0 = 1; /启动定时器while(!IR /等待低电平结束if(TF0)/若计时超过正常红外时间TF0 = 0; /清零定时器溢出标志TR0 = 0; /关闭定时器TH0 = 0;TL0 = 0;INT_CLKO |= 0x10; /使能外部中断 2return; TR0 = 0; /关闭定时器L_Time = (TH0700) /若低电平时间异常TF0 = 0; /清零定时器溢出标志TR0 = 0; /关闭定时器TH0 = 0;TL0 = 0;INT_CLKO |= 0x10; /使能外部中断 2return; /停止解码if(H_Time1400) /清零定时器溢出标志TR0 = 0; /关闭定时器TH0 = 0;TL0 = 0;INT_CLKO |= 0x10; /使能外部中断 2return; /停止解码IR_tableIR_cnt = Key_Temp;if(IR_table0+IR_table1)=0xFF) /将存储的按键值赋给 Key_TempLong_Press = 0;else Key_Temp = 0; /若验证的数据有误,则赋 0else if(L_Time8000) /长按计数INT_CLKO |= 0x10; /处理完毕,打开外部中断