数码显示(用到74ls273) |
您所在的位置:网站首页 › 74ls244控制数码管 › 数码显示(用到74ls273) |
以下程序选择的晶振都是6MHz的,实现的功能是秒和分的实时显示在数码管上,用到了6个数码管 ==>> xx--xx (左分右秒,中间是小横线) 流程图: c程序: //程序选择的晶振是6MHz #include #include #include #include #define PORT XBYTE[0x0CFA0] //定义片选地址,但是还是不知道怎么来的??? unsigned char code bySegTable[]={0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F, 0x6F,0x40};//共阴数码管的段选信号 unsigned char data byTimerTable[6]={0,0,10,10,0,0}; //要显示的时间放置在byTimerTable,其中byTimerTable[2]和 byTimerTable[3] 内容固定,是符号 “- ” unsigned char data byMinute = 0; //定义分变量,赋初值 unsigned char data bySecond = 0; //定义秒变量,赋初值 void InitTimer1(void); void Display(void); void Delay1ms(void); void main(void) { InitTimer1(); //定时/计数器 1 初始化
for(;;) { Display(); } } void InitTimer1(void) { TMOD = 0x10; //T1 工作于方式 TH1 = (65536 - 50000)/256; //置时间常数,延时 0.1 秒 TL1 = (65536 - 50000)%256; //置时间常数,延时0.1 秒 ET1 = 1; //允许 T1 中断 EA = 1; //允许总中断 TR1 = 1; //开启 T1 } void Display(void) { unsigned char data i; unsigned char data byLedSelect= 0XFE; //数码管位选置初值,准备选中第 1 个数码管 byTimerTable[4] = bySecond / 10; //获取秒的十位数 byTimerTable[5] = bySecond % 10; //获取秒的个位数 byTimerTable[0] = byMinute/ 10; //获取分钟的十位数 byTimerTable[1] = byMinute% 10; //获取分钟的个位数
for(i=0;i { PORT = bySegTable[byTimerTable[i]]; //送段选 P1 = byLedSelect; //送位选 Delay1ms(); PORT = 0x00; //关闭段选 P1 = 0xFF; //关闭位选 byLedSelect = _crol_(byLedSelect, 1); //位选左移一位(循环左移),注意和 byLedSelect byLedSelect } } void Delay1ms(void) { unsigned char a,b; for(b=71;b>0;b--) for(a=2;a>0;a--); } void Timer1Interrupt(void) interrupt 3 //T1中断服务程序 { static unsigned char data byCounter = 0; // 设置byCounter为静态局部变量
TL1 = bySecond % 10; //重置时间常数 TH1 = bySecond / 10; //重置时间常数
byCounter++; //计数器加1 if(byCounter ==10) { byCounter = 0; bySecond++; if(bySecond == 60) { bySecond = 0; byMinute++; if(byMinute== 60) { byMinute = 0; } } } } 汇编程序:(还没在实验上跑过,程序能能不实现功能不清楚,先发上来先,后若发现问题必更新,若热心人看到有错,麻烦请你提出,在此感激不尽,嘻嘻,汇编真是不好搞懂,傻傻的对着它,头都晕了) ;已在2012年5月29日7:29:44修改 PORT EQU 0CFA0H BUF EQU 23H ;存放初值 SBF EQU 22H ;存放秒值 MBF EQU 21H ; 存放分值 CSEG AT 4000H LJMP START CSEG AT 401BH LJMP CLOCK CSEG AT 4100H START: MOV R0, #40H ;40H-45H是显示缓冲区 MOV A, #00H ;依次存放高位 MOV @R0, A ;0A,0A(横线) 以及秒 INC R0 ;高位,秒低位 MOV @R0, A INC R0 MOV A, #0AH MOV @R0, A INC R0 MOV @R0, A INC R0 MOV A, #00H MOV @R0, A INC R0 MOV @R0, A MOV TMOD, #10H ;定时器1初始化为方式1 MOV TH1, #3CH ;计数50000次,即延时了50*2ms MOV TL1, #0B0H ;置时间常数,延时0.1秒 MOV BUF, #00H ;置0 MOV SBF, #00H MOV MBF, #00H SETB ET1 SETB EA SETB TR1 DS1: MOV R0, #40H ;置显示缓冲区首址 MOV R2, #0AH ;位选,置扫描初值,点亮最左边的LED6,与下面的CPL A相关 DS2: MOV DPTR, #PORT ; MOV A, @R0 ;得到的段显码输出到段数据口(即是更新的数据给A) ACALL TABLE MOVX @DPTR, A ;即是更新的数据在数码管显示 MOV A, R2 CPL A ;什么意思,要加个各个位取反干嘛?? MOV P1, A ;位选吧,选通那个数码管,但是上面的CPL A到底有何用处,与这条语句有何联系呢???【请看MOV R2, #0A】 MOV R3, #0FFH ;延时一小段时间 DEL: NOP DJNZ R3, DEL INC R0 ;显示缓冲字节加一 CLR C ;RRC与C是有联系的 MOV A, R2 RRC A ;显码右移一位,要留意于RR的区别, MOV R2, A ;最末一位是否显示完毕?,如无则继续往下显示 JNZ DS2 MOV R0, #45H ;不清楚45H填对了木有,应该对吧, MOV A, SBF ACALL GET DEC R0 DEC R0 MOV A, MBF ACALL GET SJMP DS1 TABLE: INC A ;PC指向的程序要执行的下一条语句,又一位RET占用了一个字节,若A开始是0的画,执行这条语句后,再执行MOVC A, @A+PC就指向了下面【DB 。。。】的首地址 MOVC A, @A+PC ;有很大的疑问,就是不懂,怎么就能直接求出PC的地址呢?而不是在keil调试中发现 RET DB 3FH, 06H, 5BH, 4FH, 66H, 6DH DB 7DH, 07H, 7FH, 6FH, 40H GET: MOV R1, A ;把从分或秒字节中取来的值的高 ANL A, #0F0H ;位屏蔽掉,并送入缓冲区???取高位不太清楚0F0H填的对不对 MOV @R0, A DEC R0 MOV A, R1 ;把从分或秒字节中取来的值的低 SWAP A ;位屏蔽掉,并送入缓冲区 ANL A,#0F0H ;???取低位不太清楚0F0H填的对不对 MOV @R0, A DEC R0 ;R0指针下移一位 RET CLOCK: MOV TL1, #0B0H ;置时间常数 MOV TH1, #3CH ;计数50000次,即延时了50*2ms PUSH PSW PUSH ACC INC BUF ;计数加一 CJNE A, #0AH, QUIT ;计到10否?没有则转到QUIT退出中断,注意与下面进行DA转换的区别,这里表示的十六进制数 MOV BUF, #00H MOV A, SBF INC A ;秒值加一,经十进制调整后放入 DA A ;十进制调整指令,功能是在进行BCD码加法运算时,用来对BCD码的加法运算结果进行自动修正 MOV SBF, A CJNE A, #60H, QUIT ;计到60否?没有则转到QUIT退出中断 3CH转为十进制为60,注意这里的数表示的是BCD码,与上面的DA息息相关 MOV SBF, #00H |
今日新闻 |
推荐新闻 |
CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3 |