/*-----------------------------------------------------
方案要求:上电一个LED一直闪,四位数码管9999循环正计数
思路:用两个定时器分别控制LED和数码管
----------------------------------------------------------*/
#include <reg51.h>              //51头文件
#define Uchar unsigned char      //宏定义,用Uchar来代表关键词 unsigned char(无符号字符型数据)
#define Uint unsigned int      //宏定义,用Uchar来代表关键词 unsigned int(无符号整型数据)
sbit seg = P2^6;              //位声明,声明该位用于控制数码管的笔画.
sbit com = P2^7;              //位声明,声明该位用于控制数码管的公共端.
sbit LED = P1^0;            //位声明,声明该位用于控制一个LED亮灭.
Uchar displ[]=              //声明数组,displ是自定义数组名,[]是叫下标的,里面本应填元素个数,但可以不填.
{0x3f,0x06,0x5b,0x4f,        //这些都是叫数组的元素,按顺序排放,从左到右,从上到下.
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,                    //这是共阴数码管,ox7f=01111111 ,如果该数(8字)是10000000则是共阳.
};
Uint number, number1, number2, qian, bai, shi, ge; //定义多个全局变量,以便后面要用到
void delay_mS (Uint k)      //延时子程序,k为形参,在显示子程序里会传递实参来进行计算
  {
    Uint i,j;              //定义两个局部变量
                 
    for(i=k; i>0; i--)      //k接到实参后,如果i大于0,那么让i自减1,再执行{ }内的for语句,然后再判断i是否大于0,
        {
          for(j=110; j>0; j--);
        }                        //直到i=0(不大于0),该延时子程序才算执行完成,跳出。
  }
void display (Uchar digital_4, digital_3, digital_2, Uchar digital_1)  //显示子程序, digital_2 和 digital_1 是形参,程序跑第二遍以后凡是调用该子程序的都会有实参传来
  {
      seg = 1;                    //笔画使能,74HC573的锁存控制端LE, 1(高电平)为输出跟随输入端变化而变化,0(低电平)为输出状态被锁存,不受输入端影响。
      P0 = displ[digital_1];    //调用数组,[shi]数组元素的下标,将十位数的笔画显示数据送出
      seg = 0;                    //将刚送出的笔画数据锁存输出。
      P0 = 0xff;                //该指令的作用是消影(隐),因为P0口本身也是锁存型输出,这里等于清零,让P0口全高电平(数码管全黑)
      com = 1;                    //共阴端使能,
      P0 = 0xf7;                //第一个数码管地接通点亮的数据。
      com = 0;                    //将刚送出的共阴控制数据锁存输出。
      delay_mS(5);                //调用延时子程序,将以上送出的笔画和共阴控制数据延时显示。(5)是实参,传递给延时子程序的形参。
      seg = 1;                    //下面这些指令功能同上面,只不过它送的是个位数的数据(送的是个位数数码管的显示内容)。
      P0 = displ[digital_2];
      seg = 0;
      P0 = 0xff;
      com = 1;
      P0 = 0xfb;
      com = 0;
      delay_mS(5);                  //这个延时时间也就是数码管的刷新间隔时间,可换算成刷新率Hz,如果该时间太长会有闪烁感.
      seg = 1;                   
      P0 = displ[digital_3];
      seg = 0;
      P0 = 0xff;
      com = 1;
      P0 = 0xfd;
      com = 0;
      delay_mS(5);
      seg = 1;                   
      P0 = displ[digital_4];      //这些digital可以不分先后的,因为刷新很快,人眼分辨不出哪个先后显示.
      seg = 0;
      P0 = 0xff;
      com = 1;
      P0 = 0xfe;
      com = 0;
      delay_mS(5);
  }
void main (void)                  //主程序 (函数即程序即代码)
{
      TMOD = 0x01;                //定义定时器的工作方式,TMOD是一个8位特功能寄存器,0x01=0000001,(方式1,16位定时器)
      TH0 = 0X4c;                //给定时器0装初值,高八位
      TL0 = 0Xd0;                //给定时器0装初值,低八位
      TH1 = 0X4c;                //给定时器1装初值
      TL1 = 0Xd0;
      EA = 1;                    //打开中断总允许开关,这样定时器溢出后才会进入中断服务程序
      ET0 = 1;                    //打开定时器0中断允许,这样定时器溢出后才会进入中断服务程序
      ET1 = 1;                    //打开定时器1中断允许,这样定时器溢出后才会进入中断服务程序
      TR0 = 1;                    //启动定时器0 (中断入口号为1)
      TR1 = 1;                    //启动定时器1 (中断入口号为3);两个定时器开始"同时"跑动了(实际上是TR0先跑,哪条指令在先就先执行哪条嘛)
                                //两个定时器进入的中断程序是不一样的,规定有对应的入口号
        while(1)                    //主程序循环...
动态清零是什么意思          {                        //不断调用display子程序.
            display(qian, bai, shi, ge);        //shi 和ge是变量,是实参,该实参是由中断程序传递过来,然后再传递给display子程序中的形参digital_2 和 digital_1.
          }                                    //也就是不断地循环重复运行display程序,display程序中的延时时间就构成了显示刷新率.
}
                     
void T0_time (void) interrupt 1      // 中断服务程序,服务号(中断入口号)为1 (也叫中断函数)
  {
    TH0 = 0X4c;                          //进入中断后首要任务给它重装下一次要跑的时间值(所谓重装初值)
    TL0 = 0Xd0;                        //严格来讲装初值也是要花费时间来装的,所以该方式的定时并不是很准确,如果要很准确需要用自动装初值的定时器工作方式.
    number1++;                  //每溢出进入中断一次就进行变量number1++
                   
    if(number1 == 10)              //如果加到4,说明到了所需要的延时时间,那么
      {
      number1 = 0;                  //给number赋0(清零)
      LED = ~LED;                  //给LED状态取反, 不断重复这个中断程序就得到了LED闪的效果.
      }
  }
void T1_time (void) interrupt 3      // 中断服务程序,服务号(中断入口号)为3
  {
    TH1 = 0X4c;        //初值的计算公式:例如需要20mS的定时:12X(65536-x)/11059200=0.02
S;(即20mS)12是机器周期;11059200是晶振频率;0.02是要定时的时间;65536是16位定时器最大值.
    TL1 = 0Xd0;    //0.02x11059200/12=18432 ; 65536-18432=47104 ; 47104转成十六进制=B800
    number2++;        // 那么就可写成TH1=0xB8; TL1=0x00; 提示0xB8是C51编译器里16进制的写法,不分大小写.
  if(number2 == 20)            //如果加到20,说明到了所需要的延时时间,(即0到59递增的速度)那么number++
    {
      number2 = 0;                //将其清零,以便跑第二遍程序时再重新加同样的数(定同样的时间)
      number++;                //0-59的显示数加一       
      if(number == 9999)        //显示的数(0-99),如果显示到99,那么    (两位数最多只能99了,再多就出错了)
          {
          number = 0;            //清零,也就是转为显示0(从头再来),如果这里填TR0=0; TR1=0;(关闭定时器)那么到9999停止不动.
          }
      qian = number/1000;        //千位数对1000求模
      bai = number%1000/100;        //百位数需对1000求余再对100求模
      shi = number%100/10;          //十位数,对number这个变量的当时值求模得出的数作为实参赋给主程序(68行)中的shi作为实参 (十位数)
      ge = number%10;            //个位数,对number这个变量的当时值求余得出的数作为实参赋给主程序中的ge作为实参 (个位数)
    }
  }
/***********************************
作业:
请将上面的正计数改为倒计数
************************************/