/*-----------------------------------------------------
方案要求:上电一个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; //定义多个全局变量,以便后面要用到
{
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作为实参 (个位数)
}
}
/***********************************
作业:
请将上面的正计数改为倒计数
************************************/
发布评论