4位时钟数码管中的两个闪烁点怎么控制求指教!

4位时钟数码管中的两个闪烁点怎么控制求指教!,第1张

中间那两个发光二极管不用接到动态显示电路中,可单独用一个三极管驱动来控制就好了。这两个二极管闪烁程序就好写了。

既然是时钟,肯定要定时器定时,并中断的吧。计时到半秒时,二极管控制位取反一次,就闪烁了。

没有分没有写程序动力,告诉你思路吧

1、OB35 设定1000ms循环

2、在ob35中个累加,并且安分钟和小时寄存

3、比较,当分钟的寄存器=30时,继电器输出1s

4、分钟继存器=0时,写一段循环程序:当当前小时大于临时寄存器 进行下面的程序,继电器的输出1s和一个临时寄存器的累加, 当临时寄存器的累加值大于当前小时,就不进行循环,

5、秒闪烁这个就是500ms输出,然后弄一个500ms定时器进行不输出。

就这么多了,很简单的程序,

防止窗口闪烁,不是添加什么代码的问题,而是需要优化代码。主要注意一下几个方面:

优化OnDraw中的代码,给OnDraw函数“减肥”,尽量减少每次重绘的事务,将不必要代码尽量放在只调用一次的地方。

减少对触发重绘的频繁和不合理调用。去掉不必要的Invalidate和UpdateWindow调用,使用InvalidateRect以及合理的计算区域 代替 Invalidate 这个整体重绘函数。

当重绘代码(OnDraw代码)过于耗时过于复杂的时候,应考虑使用双缓冲技术,通过内存DC优化解决闪烁问题,不过一般对于时钟程序来说,这个是没有很大必要的。

#include<reg51h>

#define uchar unsigned char

uchar code ledtab[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40};//0-9

unsigned char sec=0,min=0,hour=12,scanled;

unsigned char key,flashbit,mode,time; 

unsigned char disdat[8];

sbit led=P1^0;

void delay(unsigned int x)

{

unsigned int i,j;

for(i=0;i<x;i++)

for(j=0;j<120;j++);

}

void dischg()

{

disdat[0]=sec%10;

disdat[1]=sec/10;

disdat[2]=10;

disdat[3]=min%10;

disdat[4]=min/10;

disdat[5]=10;

disdat[6]=hour%10;

disdat[7]=hour/10;

}

void flash()

{

switch(flashbit)

{

case 0:break;

case 1:

disdat[6]=0x10;

disdat[7]=0x10;

delay(50);

dischg();

delay(80);

break;

case 2:

disdat[3]=0x10;

disdat[4]=0x10;

delay(50);

dischg();

delay(80);

break;

case 3:

disdat[0]=0x10;

disdat[1]=0x10;

delay(50);

dischg();

delay(80);

break;

default:break;

}

}

void t0isr() interrupt 1 //秒计时

{

TH0=0x3c;

TL0=0xb0;

time++;

switch(mode)

{

case 0:

if(time==20)

{

time=0;

sec++;

if(sec>59)

{

sec=0;

min++;

if(min>59)

{

min=0;

hour++;

if(hour>23)hour=0;

}

}

}

break;

case 1:

if(time==20)

{

time=0;

if(sec>0)sec--;

else if(min>0){sec=59;min--;}

else if(hour>0){sec=59;min=59;hour--;}

else {hour=0;min=0;sec=0;}

}

break;

}

dischg();

}

void t1isr() interrupt 3 //显示

{

TH1=0xec;

TL1=0x78;

switch(scanled)

{

case 0:

P2=0x01;

P0=~ledtab[disdat[7]];

break;

case 1:

P2=0x02;

P0=~ledtab[disdat[6]];

break;

case 2:

P2=0x04;

P0=~ledtab[disdat[5]];

break;

case 3:

P2=0x08;

P0=~ledtab[disdat[4]];

break;

case 4:

P2=0x10;

P0=~ledtab[disdat[3]];

break;

case 5:

P2=0x20;

P0=~ledtab[disdat[2]];

break;

case 6:

P2=0x40;

P0=~ledtab[disdat[1]];

break;

case 7:

P2=0x80;

P0=~ledtab[disdat[0]];

break;

default:break;

}

scanled++;

scanled%=8;

}

main()

{

TMOD=0x11;

TH0=0x3c;

TL0=0xb0;

TH1=0xec;

TL1=0x78;

TR1=1;

TR0=1;

ET0=1;

ET1=1;

EA=1;

sec=55;

min=59;

hour=23;

flashbit=0;

scanled=0;

time=0;

mode=0;

dischg();

while(1)

{

flash();//闪烁

if((P3&0x0f)!=0x0f){

key=P3&0x0f;

while((P3&0x0f)!=0x0f);

led=0;

delay(10);

key|=0xf0;

switch(~key)

{

case 0x01: //p31选择调时、分、秒

TR0=0;

flashbit+=1;

if(flashbit>3){flashbit=0;TR0=1;}

break;

case 0x02: //p32调数

if(flashbit==0)break;

if(flashbit==1)

{

hour++;

if(hour>99)hour=0;

}

if(flashbit==2)

{

min++;

if(min>59)min=0;

}

if(flashbit==3)

{

sec++;

if(sec>59)sec=0;

}

break;

case 0x04:  //选择正/倒

TR0=0;

mode++;

mode&=0x01;

if(mode==0){sec=0;min=0;hour=0;}

dischg();

break;

case 0x08: //启动/暂停

TR0=~TR0;

break;

default:break;

}

}

}

}

闹钟智能电子钟

源程序

源程序如下: ORG 0000H ;程序开始入口

LJMP START

ORG 0003H ;外中断0中断程序入口

RETI

ORG 000BH ;定时器T0中断程序入口

LJMP INTT0 ;跳至INTTO执行

ORG 0013H ;外中断1中断程序入口

RETI

ORG 001BH ;定时器T1中断程序入口

LJMP INTT1

ORG 0023H ;串行中断程序入口地址

RETI

;---------------主程序----------------------;

START:MOV R0,#70H ;70给R0,清70-7FH显示内存

MOV R7,#0FH

CLEARDISP:MOV @R0,#00H ;0给R0中的数为地址的内存中存放(70H)

INC R0

DJNZ R7,CLEARDISP

MOV 78H,#2 ;默认时间为12:00,闹钟5:00

MOV 79H,#1

MOV 74H,#2 ;防止上电时数码管显示00小时

MOV 75H,#1

MOV 66H,#1 ;中断退出时66H为1时,分、时计时单元数据移入显存标志

MOV 68H,#1 ;上电默认闹钟开状态

MOV 7DH,#6 ;闹钟时十位

MOV 7CH,#3

MOV 69H,#0

MOV 7AH,#0AH ;放入"熄灭符"数据

MOV TMOD,#11H ;设T0、T1为16位定时器

MOV TL0,#0B0H ;50MS定时初值(T0计时用)

MOV TH0,#3CH

SETB EA ;总中断开放

SETB ET0 ;允许T0中断

SETB TR0 ;T0定时器开始计时

MOV R4,#14H ;1秒定时用初值(50MS×20)

START1:LCALL DISPLAY ;显示子程序

LCALL BEEP ;闹钟查询

JNB P17,SETTIME ;P17口为0时转时间调整程序

JMP START1 ;P17口为1时跳回START1

;-----------------------时间闹钟调整系统--------------------------;

NFLAG:MOV A,68H ;设置闹钟开关状态

CJNE A,#1,BEE68

MOV 68H,#0

JMP E6

BEE68:MOV 68H,#1

E6:MOV 66H,#8 ;中断退出66H不为1时分、时计时单元数据移入显存标志

MOV 72H,7BH

MOV 73H,7CH

MOV 74H,7DH

MOV 75H,7EH

LCALL DDL

LCALL DDL

MOV 72H,68H

MOV 73H,69H

MOV 74H,7AH

MOV 75H,7AH

LCALL DDL

LCALL DDL

MOV 66H,#1

LJMP START1

SETTIME:LCALL DDL

JB P17,NFLAG ;键按下时间小于1秒,设置闹钟开关状态并查看闹钟时间,不关走时,确保准确,大于1秒调时

MOV TL1,#0B0H ;T1闪烁定时初值

MOV TH1,#3CH

MOV R2,#06H ;进入调时状态,赋闪烁定时初值

MOV 66H,#8 ;调闹钟时保持走时,关闭时钟显示数据

SETB ET1 ;允许T1中断

SETB TR1 ;开启定时器T1

SET1:LCALL DISPLAY ;调用显示,防止键按下无显示

JNB P17,SET1 ;P17口为0等待键释放

MOV R5,#00H ;清设置类型闪烁标志

SETN1:INC R5 ;闹钟分调整

SET5:LCALL DISPLAY

JB P17,SET5

SEETN1:LCALL DDL ;有键按下大于1秒分时间连续加(05秒加1),小于1秒转调时状态

JB P17,SET6 ;键释放查询,键释放自动转调时

MOV R0,#7CH

LCALL ADD1

MOV A,R3

CLR C

CJNE A,#60H,HHN1

HHN1:JC SEETN1

LCALL CLR0

JMP SEETN1

SET6:INC R5 ;闹钟时调整

SEET6:LCALL DISPLAY

JB P17,SEET6

SEETNH1:LCALL DDL

JB P17,SETF

MOV R0,#7EH

LCALL ADD1

MOV A,R3

CLR C

CJNE A,#24H,HOUU1

HOUU1:JC SEETNH1

LCALL CLR0

JMP SEETNH1

SETF:LCALL DISPLAY

JB P17,SETF

LCALL DDL

JNB P17,SETOUT ;短按调时,长按退出

CLR ET0

CLR TR0

MOV 70H,#00H ;设定后的时间从00秒开始走时

MOV 71H,#00H

INC R5

SET3:LCALL DISPLAY

JB P17,SET3

SETMM:LCALL DDL

JB P17,SET4

MOV R0,#77H

LCALL ADD1

MOV A,R3

CLR C

CJNE A,#60H,MMH

MMH:JC SETMM

LCALL CLR0

AJMP SETMM

SET4:INC R5

SEET4:LCALL DISPLAY

JB P17,SEET4

SETHH:LCALL DDL

JB P17,SETOUT1

MOV R0,#79H

LCALL ADD1

MOV A,R3

CLR C

CJNE A,#24H,HOUU

HOUU:JC SETHH

LCALL CLR0

AJMP SETHH

SETOUT1:SETB ET0

SETB TR0 ;计时开始

SETOUT:MOV R5,#00H ;清设置类型闪烁标志

CLR TR1 ;关闭T1

CLR ET1 ;关T1中断

MOV 66H,#1

SETOUT2:LCALL DISPLAY

JNB P17,SETOUT2

LJMP START1

;--------------------------延时1秒钟-----------------------;

DDL:MOV 18H,#36

DDL0:MOV 17H,#239

DDL1:LCALL DISPLAY

DJNZ 17H,DDL1

DJNZ 18H,DDL0

RET

;----------------------------T0中断程序------------------------;

INTT0: PUSH ACC ;打包

PUSH PSW

CLR ET0

CLR TR0

MOV A,#0B7H

ADD A,TL0

MOV TL0,A

MOV A,#3CH

ADDC A,TH0

MOV TH0,A

SETB TR0

DJNZ R4, OUTT0 ;20次中断未到中断退出

ADDSS: MOV R4,#14H ;20次中断到(1秒)重赋初值

MOV R0,#71H ;指向秒计时单元(70-71H)

ACALL ADD1 ;调用加1程序(加1秒 *** 作)

MOV A,R3 ;秒数据放入A(R3为2位十进制数组合)

CLR C ;清进位标志

CJNE A,#60H,ADDMM

ADDMM: JC OUTT01 ;小于60秒退出

ACALL CLR0 ;等于或大于60秒清0

MOV R0,#77H ;指向分计时单元(76H-77H)

ACALL ADD1

MOV A,R3

CLR C

CJNE A,#60H,ADDHH

ADDHH: JC OUTT0

ACALL CLR0

MOV R0,#79H ;指向小时计时单元(78H-79H)

ACALL ADD1

MOV A,R3

CLR C

CJNE A,#24H,HOUR

HOUR: JC OUTT0

ACALL CLR0

OUTT0: MOV A,66H ;查询标志

CJNE A,#1,OUTT01

MOV 72H,76H ;中断退出时将分、时计时单元数据移入对应显示单元

MOV 73H,77H

MOV 74H,78H

MOV 75H,79H

OUTT01:POP PSW ;出栈

POP ACC

SETB ET0 ;开放T0

RETI ;中断返回

;---------------------------加1程序--------------------------;

ADD1: MOV A,@R0 ;取当前计时单元数据到A

DEC R0 ;指向前一地址

SWAP A ;A中数据高四位与低四位交换

ORL A,@R0 ;前一地址中数据放入A中低四位

ADD A,#01H ;A加1 *** 作

DA A ;十进制调整

MOV R3,A ;移入R3寄存器

ANL A,#0FH ;高四位变0

MOV @R0,A ;放回前一地址单元

MOV A,R3 ;取回R3中暂存数据

INC R0 ;指向当前地址单元

SWAP A ;A中数据高四位与低四位交换

ANL A,#0FH ;高四位变0

MOV @R0,A ;数据放入当削地址单元中

RET

CLR0: CLR C

CLR A ;清累加器

MOV @R0,A ;清当前地址单元

DEC R0 ;指向前一地址

MOV @R0,A ;前一地址单元清0

RET ;子程序返回

;-------------------------调时闪烁程序----------------------;

INTT1: PUSH ACC

PUSH PSW

MOV TL1,#0B0H

MOV TH1,#3CH

DJNZ R2,INTT1OUT ;03秒未到退出中断(50MS中断6次)

MOV R2,#06H ;重装03秒定时用初值

CPL 02H ;03秒定时到对闪烁标志取反

JB 02H,FLASH1 ;02H位为1时显示单元"熄灭"

MOV A,R5

CJNE A,#01H,NL

JMP NNN1

NL:CJNE A,#02H,NL1

LJMP NNN1

NL1:CJNE A,#03H,NL2

LJMP NLL

NL2:CJNE A,#04H,INTT1OUT

LJMP NLL

NLL:MOV 72H,76H

MOV 73H,77H

MOV 74H,78H

MOV 75H,79H

AJMP INTT1OUT

NNN1:MOV 72H,7BH

MOV 73H,7CH

MOV 74H,7DH

MOV 75H,7EH

INTT1OUT:POP PSW ;恢复现场

POP ACC

RETI ;中断退出

FLASH1:MOV A,R5

CJNE A,#01H,ML

LJMP MNN1

ML:CJNE A,#02H,ML1

LJMP MNN2

ML1:CJNE A,#03H,MN

LJMP MLL

MN:CJNE A,#04H,INTT1OUT

LJMP MHL

MLL:MOV 72H,7AH

MOV 73H,7AH ;显示单元(72-73H),将不显示分数据

MOV 74H,78H

MOV 75H,79H

AJMP INTT1OUT

MHL:MOV 72H,76H

MOV 73H,77H

MOV 74H,7AH ;显示单元(74-75H),小时数据将不显示

MOV 75H,7AH

AJMP INTT1OUT

MNN1:MOV 72H,7AH

MOV 73H,7AH ;显示单元(72-73H),将不显示闹钟分数据

MOV 74H,7DH

MOV 75H,7EH

AJMP INTT1OUT

MNN2:MOV 72H,7BH

MOV 73H,7CH

MOV 74H,7AH

MOV 75H,7AH ;显示单元(74-75H),闹钟小时数据将不显示

AJMP INTT1OUT

;-------------------------显示-------------------------------;

DISPLAY:MOV DPTR,#DISDATA

MOV A,70H

MOVC A,@A+DPTR

MOV P3,A

CLR P10

NOP

NOP

NOP

SETB P10

MOV A,71H

MOVC A,@A+DPTR

MOV P3,A

CLR P11

NOP

NOP

NOP

SETB P11

MOV A,72H

MOVC A,@A+DPTR

MOV P3,A

CLR P12

NOP

NOP

查询闹钟分十位

CJNE A,7CH,BEERE

BB:CLR P16

LCALL DDL

SETB P16

LCALL DDL

JNB P17,BEERR

BEERE:RET

BEERR:SETB P16

MOV 68H,#0

LCALL DISPLAY

JNB P17,BEERR

JMP BEERE

DISDATA:DB 0C0H,0F9H,64H,70H,59H,52H,42H,0F8H,40H,50H,0FFH ;0123456789空白

END

这个是与编程思想有关,你按键用的消抖方式是延时消抖,不是用状态机的编程方式,所以会如此

参考一下电路及代码

#include<reg51h>

#include<INTRINSH>

#define Key_State_0 0

#define Key_State_1 1

#define Key_State_2 2

#define unlock 0

#define lock   1

#define Init_DATA 999

unsigned char key1_state;

bit key1_debounce_flag;

bit key1_debounce;

sbit Key_1 = P1^0;

sbit seg1  = P1^7;

sbit seg2  = P1^6;

sbit seg3  = P1^5;

sbit seg4  = P1^4;

unsigned char time_1ms_flag,display_flag;

unsigned int  add_data;

unsigned char key1_debounce_C;

unsigned char code table[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};

void MCU_init(void)

{

TMOD = 0x01;

TH0  = 0xFC;

TL0  = 0x18;

EA   = 1;

ET0  = 1;

TR0  = 1;

add_data = Init_DATA;

}

void MCU_clock(void)

{

if(time_1ms_flag==1)

{

time_1ms_flag = 0;

display_flag++;

if(display_flag == 4)

{

display_flag = 0;

}

if(key1_debounce_flag==unlock)

{

key1_debounce_C++;

if(key1_debounce_C==20)

{

key1_debounce_C = 0;

key1_debounce_flag = lock;

key1_debounce = 1;

}

}

}

}

void display(void)

{

switch(display_flag)

{

case 0:

seg4 = 0;

P2=table[add_data/1000];

seg1 = 1;

break;

case 1:

seg1 = 0;

P2=table[add_data%1000/100];

seg2 = 1;

break;

case 2:

seg2 = 0;

P2=table[add_data%100/10];

seg3 = 1;

break;

case 3:

seg3 = 0;

P2=table[add_data%10];

seg4 = 1;

break;

}

}

void Key_1_Scan (void)

{

bit key_press;

key_press = Key_1;

switch(key1_state)

{

case Key_State_0:

if(!key_press)

{

key1_state = Key_State_1;

key1_debounce_flag = unlock;

}

break;

case Key_State_1:

if(key1_debounce==1)

{

key1_debounce = 0;

if(!key_press)

{

key1_state = Key_State_2;

}

else

{

key1_state = Key_State_0;

}

}

break;

case Key_State_2:

if(key_press)

{

key1_state = Key_State_0;

add_data++;

if(add_data==10000)

{

add_data = 0;

}

}

break;

}

}

void main (void)

{

MCU_init();

while (1)

{

MCU_clock();

Key_1_Scan();

display();

}

}

void Timer0Interrupt(void) interrupt 1

{

TH0 = 0xFC;

TL0 = 0x18;

time_1ms_flag = 1;

}

在硬件组态界面,双击cpu,里面有个Cycle/Clock Memory选项,最下面的Clock Memory 勾选中,配置一个字节,比如你配置个10,那程序里的M100到M107就代表不同的频率,可以实现闪烁。

以上就是关于4位时钟数码管中的两个闪烁点怎么控制求指教!全部的内容,包括:4位时钟数码管中的两个闪烁点怎么控制求指教!、用plc编程24小时时钟程序、怎样在时钟程序中添加防止屏幕闪烁的代码,vc++ondraw函数中添加等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zz/10081611.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-05
下一篇 2023-05-05

发表评论

登录后才能评论

评论列表(0条)

保存