一、引言
在FB程序开发中,定时器是常见的功能之一,用于控制程序的执行流程,实现延时操作等。
在使用定时器的过程中,可能会遇到各种问题和异常。
本文将介绍FB程序中定时器的应用技巧以及解决程序异常的方法。
二、FB程序中定时器的应用技巧
1. 定时器类型选择
FB程序中常用的定时器类型包括单次触发定时器和周期触发定时器。
单次触发定时器在达到设定时间后只触发一次,适用于单次延时操作;周期触发定时器每隔一定时间触发一次,适用于周期性执行的任务。
根据实际需求选择合适的定时器类型,可以提高程序的效率和稳定性。
2. 定时器的启动与停止
在FB程序中,可以通过调用定时器函数来启动和停止定时器。
启动定时器时,需要设置定时器的时间参数和触发条件;停止定时器时,需要及时清除定时器,避免资源浪费和意外触发。
3. 定时器的精度与调整
定时器的精度直接影响到程序的执行效率。
在选择定时器时,需要考虑其精度是否满足需求。
若定时器的精度不够,可能导致程序执行时出现误差。
可以通过优化程序结构、减少不必要的操作等方式提高定时器的精度。
三、解决程序异常的方法
1. 异常类型识别
在FB程序中,常见的异常类型包括运行时错误、语法错误和逻辑错误等。
运行时错误通常是由于程序运行过程中出现的异常情况,如空指针访问、数组越界等;语法错误是由于编程语法不正确导致的错误;逻辑错误是由于程序逻辑设计不合理导致的错误。
识别异常类型对于解决问题至关重要。
2. 异常处理策略
针对不同类型的异常,需要采取不同的处理策略。
对于运行时错误,可以通过异常捕获机制来捕获并处理异常情况;对于语法错误,需要仔细检查代码,修正语法错误;对于逻辑错误,需要优化程序逻辑,确保程序的正确执行。
3. 错误调试与日志记录
在FB程序中,可以使用调试工具进行错误调试,找出错误发生的位置和原因。
同时,通过日志记录的方式,可以实时记录程序的运行情况和异常信息,方便问题的定位和解决。
四、结合实例分析定时器应用与异常处理
假设在FB程序中,我们需要实现一个定时提醒功能,每隔一定时间发送提醒消息。
我们可以使用周期触发定时器来实现这个功能。
在定时器触发时,执行发送提醒消息的操作。
在实际使用过程中,可能会因为网络问题或其他原因导致提醒消息无法发送。
这时,我们需要进行异常处理,确保程序的稳定运行。
我们可以使用异常捕获机制来捕获发送消息过程中的异常情况,并进行相应处理。
同时,通过日志记录的方式,记录提醒消息的发送情况和异常信息。
如果定时器触发后长时间未收到消息发送成功的反馈,我们可以重新启动定时器,或者采取其他措施来确保提醒功能的正常运行。
五、总结
本文介绍了FB程序中定时器的应用技巧以及解决程序异常的方法。
通过选择合适的定时器类型、控制定时器的启动与停止、提高定时器的精度等技巧,可以更有效地使用定时器。
同时,通过识别异常类型、采取异常处理策略、进行错误调试与日志记录等方法,可以提高程序的稳定性和可靠性。
在实际开发中,我们需要结合具体需求和应用场景,灵活运用这些技巧和方法,实现高效、稳定的FB程序。
我要用 89c52单片机做 一个 时钟程序 起始时间是0:00,但是 发现 定时器1的中断程序一直都不执行?
节省一点资源,只用一个定时器就够了
而且采用自装入方式定时,可以减少计时误差
试试:
单片机定时器程序仿真问题
在主程序里除了初始化的那个函数外把剩下的语句写到while(1){}里面不加死循环当然就只执行一次了
数码管程序。前一个程序和后一个程序在定时器设定及以后的程序内容有区别。请大神给出为啥第一个程序错误
/* 正确的如下:还有一个问题就是触发外部中断后进入第三状态时,怎么退出来,进入中断后第1状态:每隔1秒加1,加到100数,跳到第2状态:8个数码管各自点亮口字,循环5次后进入第3状态:8个数码管组成的大口子,循环5次后,让其再回到第1状态这样循环着除非单片机复位,现在问题停在第3状态循环出不来,没有可调用的涵数,待学习请助中,请高手们指点一下迷区*/#include<reg52.h>//载入51单片机头文件#include<intrins.h>//载入51涵数头文件#define uchar unsigned char //宏定义unsigned char为 uchar#define uint unsigned int//宏定义unsigned int 为uintsbit wela=P3^7;//位定义,控制数码管有几个数量的373锁存器端口接在该I/Osbit dula=P2^0;//位定义,控制数码管显示什么字符的373锁存器端口接在该I/Osbit kk=P3^2;uchar k,j,temp,led,dd,tt,qian,bai,shi,ge,rr,yj,ee;//单字节全局变量uint ptemp;//双字节全局变量uchar code dutable[]={//定义数码管显示,code为表格的意思,该表格取名为dutable0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90//根据数码管硬件连接图写出0-9的16进制数};uchar code wetable[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};//根据数码管硬件连接图写出1-8个数码管亮的位//…………以下延时程序…………void delay(uint z)// 这里为什么用uint(双字节),而不是用uchar(单字节,值范围256){uint x,y;for(x=z;x>0;x--)for(y=1000;y>0;y--);} //…………以上延时程序………… //………………子程序……………………void zuchngxiu(){ if(dd==1) /*这里进入定时器次和触发定时器的时间一起控制显示速度,现在这样就属于动太扫描*/{ dd=0; /*如果进入到20次定时器,那么将定时器里的变量dd清0,便于下一次计时*/tt++;//用于外部中断时间判断//……………………………………P1=0x00;//P1口全部送低电平,目的是将这低电平保存在点阵的高电平脚上,使用其不亮;P3=0x07;/*P3口关闭高位开关()其中有三个脚接点阵高点位的锁存端,其中还有外部中断*/P2=0x00; /*P2口接了三种类型,点阵低电平的锁存端5个,点阵高点位的锁存端2个,还一个是控制8个数码管段选的锁存端,这样点阵高电平脚全部由锁器端锁存为低电平了*///……………关闭点阵…………………wela=1; //打开位选P1=wetable[j]; //P1口取wetable位码表wela=0;//关闭位选j++; //每取一次位码表自加1if(j==8) //检测j是否等于9j=0;//如果是等于9就置0dula=1;//打开段选,是显示数字P1=dutable[k]; //P1端口取dutable数码表中值,dula=0; //关闭段选 k++; //k自加1,如果不自加1怎么知道它现在取的是第几位数了if(k>=9)/*检测k是否大于等于9,因为要显示的数是1-8的数*/k=1; /*如果上面K大于等于9,就将K清1 ,而不是将K清0,这样做的目的是让其显示从1数开始显示9位数(1-8),如果清0,那就会从0开始显数*/}}//……………………………………//………………以下是累加显示子程序…………………………void display(uchar qian,uchar bai,uchar shi,uchar ge){ wela=1;P1=0x01;wela=0;dula=1;P1=dutable[qian];dula=0; delay(1);//………………………………wela=1;P1=0x02;wela=0;dula=1;P1=dutable[bai];dula=0;delay(1);//……………………………… wela=1;P1=0x04;wela=0;dula=1;P1=dutable[shi];dula=0;delay(1);//……………………………… wela=1;P1=0x08;wela=0;dula=1;P1=dutable[ge];dula=0;delay(1);}//………………以上是累加显示子程序………………//……………………以下循环8位数码管组成的口子子程序…………………………voidxunhuankou(){dula=1; //打开段选,P1=0xfe; //显示a段,这时8个数码管都显示a段的,所以在这前要先选好位dula=0;delay(10);wela=1; P1=wetable[yj]; //P1口取wetable位码表 也就是第一们数码管delay(3);P1=0x02;delay(3);P1=0x04;delay(3);P1=0x08;delay(3);P1=0x10; delay(3);P1=0x20;delay(3);P1=0x40;delay(3);P1=0x80;delay(3);wela=0; //…………………… dula=1;P1=0xfd; delay(3);P1=0xfb;delay(3);P1=0xf7;dula=0;delay(3);//……………………wela=1;P1=0x80;delay(3);P1=0x40;delay(10);P1=0x20;delay(3);P1=0x10;delay(3);P1=0x08;delay(3);P1=0x04;delay(3);P1=0x02;delay(3);P1=0x01;wela=0;delay(3);//…………………………dula=1;P1= 0xef;delay(3);P1= 0xdf;delay(3); dula=0;//………………………………}//……………………以上循环8位数码管组成的口子子程序………………………………//……………………以下初始化程序……………………void init(){kk=1; //给外部中断0由软件置高电平ptemp=0; //给累加器一个初值EA=1; //开总中断ET0=1;//定时器/计数器0中断TMOD=0x01; //设为定时器,单触发,01工作方式;TH0=(-5000)/256; //装高位初值TL0=(-5000)%256;//装低位初值TR0=1;//启动定时器} //…………………以上初始化程序………………………//……………………以下是外部中断中循环子程序……………………………………void zhongduan(){ init();//初始化程序ptemp=0;//给加累数初值0while(1){if(tt>=1)//判断tt是否大于等于1,这里tt就是主程序里dd=20,tt自加1tt=0; //置0重新计时ptemp++;//初值自加1if(ptemp==101) //给累加数定一个最大数 {ptemp=0;if(ptemp==0){ wela=1;//打开位选P1=0xff; //将P1口置数全部高电平, 打开8个数码管wela=0;//关闭位选dula=1;//打开段选temp=0xfe; //赋temp值为 ,也就是数码管的a段P1=temp; //将temp的值赋予给P1口delay(10); //调用延时程序while(1){temp=_crol_(temp,1);//左移一位P1=temp; //移位后的值赋予给P1口delay(8); //调用延碧rr++;//自加1,便于下面调用(rr=6刚好循环一次口字)if(temp==0xdf) /*检测temp是否等于(高位低位),因为数码管七段加一个DP点我们不要DP(P1.7)点和G段(P1.6)亮*/temp=0x7f; //置位到DP(P1.7)点,也就是P1.6和P1.7跳过(高位0111 1111低位)if(rr>=30)//就是循环5个口字 { while(1) { xunhuankou();//调用循环8位数码管组成的口子子程序 ee++;//每循环一次自加1//……………………………………………………………………………………//…………………有待解决小循环里的死循环…………………………………//if(ee==5)//让8位数码管组成的口子循环5次//问题来了,到5次后没有返回函数了,待解决(最好还是返回到外部中断一开始的地方)//………………………………………………………………………………………………//………………………………………………………………………………………………}/// }} } } qian=ptemp/1000;bai=ptemp%1000/100;shi=ptemp%1000%100/10;ge=ptemp%10;display(qian,bai,shi,ge);} } //……………………以上是外部中断中循环子程序……………………………………//………………………以下主程序……………………………void main(){ init();//初始化子程序EX0=1;//开外部中断0IT0=0;//外部中断0以电平方式触发PX0=0;//设外部中断0为最低优先级PT0=1;//设该定时器为最高级优先级P2=0x00;//程序入口就关闭点阵锁存端,要是不关闭一会数码管语句时会让点阵也显示着内容P3=0x04;//给P3口除外部中断0高点平外,都低电平。 外部中断0是低平电有效的wela=1;//打开控制数码管工作的373端口,输入高电平有效,相当于输入与输出端直通P1=0xff; //送入让那几个数码管亮的16进制数wela=0;//关闭控制数码管工作373端口,这样373就保持住输入端上一次输入的值dula=1;//打开控制数码管显示什么数字的373端口P1=0x00;/*输入让数码管显示什么数,这里是全部亮就是日字,作为检测一下数码管 有无坏点*/dula=0;//关闭控制数码管显示什么数字的373端口 P1=0xff;//P1送入高电平P2=0x3e; //打开低位开关P1=0x00; //P1送入低电平P2=0xc0;//关闭位开关 P3=0x6f;//打开高位开关P1=0xff;//P1送入高电平led=P0=0xfe;//这里给led赋个初值这样下面定时器里led左移才知道从什么数开始移delay(15); //这里加延时程序,某值是时段肯定会和其它地方冲突造成不定时延时现象//调用延时子程序j=0;//给初值0k=1;/*给初值1,这里为什么不是给初值0,因为k段码表也就是显示数字的表,我们现在让程序1开始显,而utable表第一位数0xc0是数为0数,*/while(1) {zuchngxiu(); //调用子程序}}// ……………………定时器0………………………………void exter1() interrupt 1 /*exter1是中断名,后面1是单片机中断中该定时器/计数器0的中断序号 */ { TH0=(-5000)/256;//求抹,将整数部分装入高8位中 TL0=(-5000)%256; //求余,将余数部分装入低8位中 //这里的就是50ms(毫秒) dd++;// 这里的dd时50ms触发一次中断,每到50ms计一次数,便于其它调用 led=_crol_(led,1); //将上面的led值进入左移1位 P0=led;//左移一位得到的值,赋予给P0口 }//…………………………这里定时器中断优先等级比外部中断0高…………//……………触发外部中断0后始终循环…………void exter0() interrupt 0/*exter0是断名,后面0是单片机中断中该外部中断0的中断序号 */ {zhongduan();//调用外部中断循环子程序}
本文原创来源:电气TV网,欢迎收藏本网址,收藏不迷路哦!
添加新评论