四位数码管(动态) - C51程序详解

  • 格式:docx
  • 大小:19.85 KB
  • 文档页数:4

下载文档原格式

  / 13
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

/*-----------------------------------------------------

方案要求:上电一个LED一直闪,四位数码管9999循环正计数

思路:用两个定时器分别控制LED和数码管

----------------------------------------------------------*/

#include //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,说明到了所需要的延时时间,那么

{