请教大哥大姐们51单片机一个定时器,多处程序怎么用

2019-07-15 23:03发布

一个定时器,多个程序用,(如主程序中让每进20次定时中断(1秒),让灯亮一下,这样循环着,当触发外部中断时,可以是1秒也可以大于1秒,响应另一个触发事件)
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
11条回答
那风又起
1楼-- · 2019-07-16 00:49
你可以弄一个变量假设是N,每中断一次对N计数,根据N的数值来判断执行那个程序
auqyygtj
2楼-- · 2019-07-16 03:56
本帖最后由 auqyygtj 于 2013-8-13 22:09 编辑

大家帮我分析一下问题如下:1、现在是主程序动显在那0至7,我要的是1-8动显在那;2、触发外部中断后,加数器不动,可能是一个定时器,两个地方用,什么地方没用正确,外部中断的部分计时器没起作用;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
uchar k,j,temp,led,dd,tt,qian,bai,shi,ge;
//单字节全局变量
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++;//用于外部中断时间判断
                                if(j==8) //检测j是否等于9
                                j=0;  //如果是等于9就置0
//……………………………………
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(k>=8)  /*检测k是否大于等于11,而不是k=10(0-9)数显示,为会下面会用到*/
                                k=0;        //如果上面K大于等于11,就将K清0 ,这样做的目的是让其显示0-9的10个数,不然后一直加上去了
                                dula=1;         //打开段选,是显示数字
                                P1=dutable[k]; //P1端口取dutable数码表中值,
                                dula=0;                   //关闭段选         
                                k++; //k自加1,如果不自加1怎么知道它现在取的是第几位数了
                }         
}
//……………………………………


//………………以下是累加显示子程序…………………………
void display(uchar qian,uchar bai,uchar shi,uchar ge)
{
   wela=1;
P1=0x01;
wela=0;
  dula=1;
P1=dutable[qian];
dula=0;
delay(5);
//………………………………  

  wela=1;
P1=0x02;
wela=0;
  dula=1;
P1=dutable[bai];
dula=0;
  delay(15);
//………………………………
   wela=1;
P1=0x04;
wela=0;
  dula=1;
P1=dutable[shi];
dula=0;
  delay(15);

//………………………………
   wela=1;
P1=0x08;
wela=0;
  dula=1;
P1=dutable[ge];
dula=0;
  delay(5);

}
//………………以上是累加显示子程序………………
//……………………以下初始化程序……………………
void init()
{
ptemp=0; //给累加器一个初值
EA=1; //开总中断
ET0=1;//定时器/计数器0中断
TMOD=0x01; //设为定时器,单触发,01工作方式;
TH0=(65536-5000)/256; //装高位初值
TL0=(65536-5000)%256;  //装低位初值
TR0=1;//启动定时器



}
//…………………以上初始化程序………………………

//………………………以下主程序……………………………

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);        //这里加延时程序,某值是时段肯定会和其它地方冲突造成不定时延时现象
                //调用延时子程序
        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的中断序号 */
{  //1

        init();
        ptemp=0;  //给加累数初值0
        while(1)
                {  //2
                if(tt>=1)//判断tt是否大于等于1,这里tt就是主程序里dd=20,tt自加1
                {  //3
                tt=0; //置0重新计时
                ptemp++;//初值自加1
                if(ptemp==10000)
                {  //4
                ptemp=0;
                if(ptemp==0)
                { //5
                wela=1;  //打开位选
                P1=0xff; //将P1口置数11111111全部高电平, 打开8个数码管
                wela=0;  //关闭位选
                dula=1;//打开段选
                temp=0xfe;        //赋temp值为11111110 ,也就是数码管的a段
                P1=temp; //将temp的值赋予给P1口
                delay(10);        //调用延时程序
                while(1)
                {         //6
                if(temp==0xdf) /*检测temp是否等于(高位11011111低位),因为数码管七段加一个DP点
                我们不要DP(P1.7)点和G段(P1.6)亮*/
                temp=0x7f; //置位到DP(P1.7)点,也就是P1.6和P1.7跳过(高位0111 1111低位)
                temp=_crol_(temp,1);  //左移一位
                P1=temp;          //移位后的值赋予给P1口
                delay(8);          //调用延碧
                }        //6
                qian=temp/1000;
                bai=temp%1000/100;
                shi=temp%1000%100/10;
                ge=temp%10;
                display(qian,bai,shi,ge);
                }
                }
                }                        
                }
}


111.jpg
jeddy1992
3楼-- · 2019-07-16 04:03
 精彩回答 2  元偷偷看……
西雅图之恋
4楼-- · 2019-07-16 09:31
3楼说的对 中断里面 不应该有while(1)语句吧 要不就该无线循环了
auqyygtj
5楼-- · 2019-07-16 10:45
找到问题了,还加了一小部分其它循环
auqyygtj
6楼-- · 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();//调用外部中断循环子程序

}

一周热门 更多>