四位数码管(动态) - C51程序详解
- 格式:docx
- 大小:19.85 KB
- 文档页数:4
/*-----------------------------------------------------
方案要求:上电一个LED一直闪,四位数码管9999循环正计数
思路:用两个定时器分别控制LED和数码管
----------------------------------------------------------*/
#include
#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,说明到了所需要的延时时间,那么
{