专家
公告
财富商城
电子网
旗下网站
首页
问题库
专栏
标签库
话题
专家
NEW
门户
发布
提问题
发文章
51单片机
请教大哥大姐们51单片机一个定时器,多处程序怎么用
2019-07-15 23:03
发布
×
打开微信“扫一扫”,打开网页后点击屏幕右上角分享按钮
站内问答
/
51单片机
8915
11
972
一个定时器,多个程序用,(如主程序中让每进20次定时中断(1秒),让灯亮一下,这样循环着,当触发外部中断时,可以是1秒也可以大于1秒,响应另一个触发事件)
友情提示:
此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
11条回答
auqyygtj
2019-07-16 11:35
本帖最后由 auqyygtj 于 2013-8-14 13:38 编辑
/* 正确的如下:还有一个问题就是触发外部中断后进入第三状态时,怎么退出来,
进入中断后第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 为uint
sbit wela=P3^7;
//位定义,控制数码管有几个数量的373锁存器端口接在该I/O
sbit dula=P2^0;
//位定义,控制数码管显示什么字符的373锁存器端口接在该I/O
sbit kk=P3^2;
uchar k,j,temp,led,dd,tt,qian,bai,shi,ge,rr,yj,ee;
//单字节全局变量
uint ptemp;
//双字节全局变量
uchar code dutable[]={
//定义数码管显示,code为表格的意思,该表格取名为dutable
0xc0,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口关闭高位开关(00000111)其中有三个脚接点阵高点位的锁存端,其中还有外部中断*/
P2=0x00; /*P2口接了三种类型,点阵低电平的锁存端5个,点阵高点位的锁存端2个,还一个是控制
8个数码管段选的锁存端,这样点阵高电平脚全部由锁器端锁存为低电平了*/
//……………关闭点阵…………………
wela=1; //打开位选
P1=wetable[j]; //P1口取wetable位码表
wela=0;//关闭位选
j++; //每取一次位码表自加1
if(j==8) //检测j是否等于9
j=0; //如果是等于9就置0
dula=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位数码管组成的口子子程序…………………………
void xunhuankou()
{
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=(65536-5000)/256; //装高位初值
TL0=(65536-5000)%256; //装低位初值
TR0=1;//启动定时器
}
//…………………以上初始化程序………………………
//……………………以下是外部中断中循环子程序……………………………………
void zhongduan()
{
init();//初始化程序
ptemp=0; //给加累数初值0
while(1)
{
if(tt>=1)//判断tt是否大于等于1,这里tt就是主程序里dd=20,tt自加1
tt=0; //置0重新计时
ptemp++;//初值自加1
if(ptemp==101) //给累加数定一个最大数
{
ptemp=0;
if(ptemp==0)
{
wela=1; //打开位选
P1=0xff; //将P1口置数11111111全部高电平, 打开8个数码管
wela=0; //关闭位选
dula=1;//打开段选
temp=0xfe; //赋temp值为11111110 ,也就是数码管的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是否等于(高位11011111低位),因为数码管七段加一个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;//开外部中断0
IT0=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; //给初值0
k=1; /*给初值1,这里为什么不是给初值0,因为k段码表也就是显示数字的表,我们现在
让程序1开始显,而utable表第一位数0xc0是数为0数,*/
while(1)
{
zuchngxiu(); //调用子程序
}
}
// ……………………定时器0………………………………
void exter1() interrupt 1 /*exter1是中断名,后面1是单片机中断中该定时器/计数器0的中断序号 */
{
TH0=(65536-5000)/256;//求抹,将整数部分装入高8位中
TL0=(65536-5000)%256; //求余,将余数部分装入低8位中
//这里的50000就是50ms(毫秒)
dd++; // 这里的dd时50ms触发一次中断,每到50ms计一次数,便于其它调用
led=_crol_(led,1); //将上面的led值进入左移1位
P0=led; //左移一位得到的值,赋予给P0口
}
//…………………………这里定时器中断优先等级比外部中断0高…………
//……………触发外部中断0后始终循环…………
void exter0() interrupt 0 /*exter0是断名,后面0是单片机中断中该外部中断0的中断序号 */
{
zhongduan();//调用外部中断循环子程序
}
加载中...
查看其它11个回答
一周热门
更多
>
相关问题
【东软载波ESF0654 PDS开发板活动】开箱
1 个回答
东软载波ESF0654 PDS开发板外部中断
1 个回答
东软载波ESF0654 PDS开发板高级控制定时器AD16C4T
1 个回答
用串口调试助手为什么只能在hex模式接收发送而在文本模式不行
9 个回答
触摸芯片SC02B/SC04B在地砖灯的设计方案
1 个回答
相关文章
51单片机与蓝牙模块连接
0个评论
51单片机的硬件结构
0个评论
基于51单片机的无线遥控器制作
0个评论
51单片机 AD转换
0个评论
51单片机数码管递增显示
0个评论
如何实现对单片机寄存器的访问
0个评论
基于51单片机的指纹密码锁
0个评论
×
关闭
采纳回答
向帮助了您的知道网友说句感谢的话吧!
非常感谢!
确 认
×
关闭
编辑标签
最多设置5个标签!
51单片机
保存
关闭
×
关闭
举报内容
检举类型
检举内容
检举用户
检举原因
广告推广
恶意灌水
回答内容与提问无关
抄袭答案
其他
检举说明(必填)
提交
关闭
×
打开微信“扫一扫”,打开网页后点击屏幕右上角分享按钮
×
付费偷看金额在0.1-10元之间
确定
×
关闭
您已邀请
0
人回答
查看邀请
擅长该话题的人
回答过该话题的人
我关注的人
/* 正确的如下:还有一个问题就是触发外部中断后进入第三状态时,怎么退出来,
进入中断后第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 为uint
sbit wela=P3^7;
//位定义,控制数码管有几个数量的373锁存器端口接在该I/O
sbit dula=P2^0;
//位定义,控制数码管显示什么字符的373锁存器端口接在该I/O
sbit kk=P3^2;
uchar k,j,temp,led,dd,tt,qian,bai,shi,ge,rr,yj,ee;
//单字节全局变量
uint ptemp;
//双字节全局变量
uchar code dutable[]={
//定义数码管显示,code为表格的意思,该表格取名为dutable
0xc0,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口关闭高位开关(00000111)其中有三个脚接点阵高点位的锁存端,其中还有外部中断*/
P2=0x00; /*P2口接了三种类型,点阵低电平的锁存端5个,点阵高点位的锁存端2个,还一个是控制
8个数码管段选的锁存端,这样点阵高电平脚全部由锁器端锁存为低电平了*/
//……………关闭点阵…………………
wela=1; //打开位选
P1=wetable[j]; //P1口取wetable位码表
wela=0;//关闭位选
j++; //每取一次位码表自加1
if(j==8) //检测j是否等于9
j=0; //如果是等于9就置0
dula=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位数码管组成的口子子程序…………………………
void xunhuankou()
{
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=(65536-5000)/256; //装高位初值
TL0=(65536-5000)%256; //装低位初值
TR0=1;//启动定时器
}
//…………………以上初始化程序………………………
//……………………以下是外部中断中循环子程序……………………………………
void zhongduan()
{
init();//初始化程序
ptemp=0; //给加累数初值0
while(1)
{
if(tt>=1)//判断tt是否大于等于1,这里tt就是主程序里dd=20,tt自加1
tt=0; //置0重新计时
ptemp++;//初值自加1
if(ptemp==101) //给累加数定一个最大数
{
ptemp=0;
if(ptemp==0)
{
wela=1; //打开位选
P1=0xff; //将P1口置数11111111全部高电平, 打开8个数码管
wela=0; //关闭位选
dula=1;//打开段选
temp=0xfe; //赋temp值为11111110 ,也就是数码管的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是否等于(高位11011111低位),因为数码管七段加一个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;//开外部中断0
IT0=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; //给初值0
k=1; /*给初值1,这里为什么不是给初值0,因为k段码表也就是显示数字的表,我们现在
让程序1开始显,而utable表第一位数0xc0是数为0数,*/
while(1)
{
zuchngxiu(); //调用子程序
}
}
// ……………………定时器0………………………………
void exter1() interrupt 1 /*exter1是中断名,后面1是单片机中断中该定时器/计数器0的中断序号 */
{
TH0=(65536-5000)/256;//求抹,将整数部分装入高8位中
TL0=(65536-5000)%256; //求余,将余数部分装入低8位中
//这里的50000就是50ms(毫秒)
dd++; // 这里的dd时50ms触发一次中断,每到50ms计一次数,便于其它调用
led=_crol_(led,1); //将上面的led值进入左移1位
P0=led; //左移一位得到的值,赋予给P0口
}
//…………………………这里定时器中断优先等级比外部中断0高…………
//……………触发外部中断0后始终循环…………
void exter0() interrupt 0 /*exter0是断名,后面0是单片机中断中该外部中断0的中断序号 */
{
zhongduan();//调用外部中断循环子程序
}
一周热门 更多>