① 如何做一个单片机电子时钟
这个很简单啊,两种思路:1,用单片机定时器来做,优点是外围电路简单,只需要一个单片机最小系统,和一个显示模块(1602液晶比较好);缺点是:定时不是很准,跑的时间长了会有误差,而且主要是程序,要写很多,不过程序不是很难。2,用时钟芯片,一般都用DS1302,还需要单片机最小系统和显示模块,优点是:定时准确,误差很小,在有备用电池的情况下,即使单片机掉电,也可以维持很长时间,程序相对简单。缺点是:外围电路比前者相对复杂一点。如果想做个实际的东西拿来用,建议使用第二种方法,如果是想锻炼自己的编程能力的话,选第一种吧。祝成功!
② 单片机实现电子时钟
ORG 0000H ;程序执行开始地址
LJMP START ;跳到标号START执行
ORG 0003H ;外中断0中断程序入口
LJMP OUTINTT0 ;外中断0中断返回
ORG 000BH ;定时器T0中断程序入口
LJMP INTT0 ;跳至INTTO执行
ORG 0013H ;外中断1中断程序入口
RETI;外中断1中断返回
ORG 001BH ;定时器T1中断程序入口
LJMP INTT1 ;跳至INTT1执行;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 主 程 序 ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
START:
MOV R0,#70H ;清70H-7AH共11个内存单元
MOV R7,#0BH ;
CLEARDISP:
MOV @R0,#00H ;
INC R0 ;
DJNZ R7,CLEARDISP ;
MOV 70H,#07H;
MOV 71H,#05H;
MOV 76H,#09H;
MOV 77H,#05H;
MOV 78H,#05H;
MOV 79H,#00H;
MOV 20H,#00H ;清20H(标志用)
MOV 7AH,#0AH ;放入"熄灭符"数据
MOV 7BH,#00H ;清报时寄存器
MOV TMOD,#11H ;设T0、T1为16位定时器
MOV TL0,#0B0H ;50MS定时初值(T0计时用)
MOV TH0,#3CH ;50MS定时初值
MOV TL1,#0B0H ;50MS定时初值(T1闪烁定时用)
MOV TH1,#3CH ;50MS定时初值
SETB PT1;定时器T1为高优先级
SETB EA ;总中断开放
SETB ET0 ;允许T0中断
SETB TR0 ;开启T0定时器
SETB EX0 ;开启外部中断0
MOV R4,#14H ;1秒定时用初值(50MS×20)
MOV R2,#06H ;0.3秒的闪动初值(50MS*6)
START1:
LCALL DISPLAY;调用显示子程序
LCALL BELL;
LCALL DISCLOSE;
SJMP START1 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 计时程序 ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;T0中断服务程序
INTT0:
PUSH ACC ;累加器入栈保护
PUSH PSW ;状态字入栈保护
CLR ET0 ;关T0中断允许
CLR TR0 ;关闭定时器T0
MOV TL0,#0B0H ;重装初值(低8位修正值)
MOV TH0,#3CH ;重装初值(高8位修正值)
SETB TR0 ;开启定时器T0
DJNZ R4, OUTT0 ;20次中断未到中断退出
ADDSS:
MOV R4,#14H ;20次中断到(1秒)重赋初值
MOV R0,#71H ;指向秒计时单元(70H-71H)
ACALL DBchange ;调用数据处理程序(加1并存入显示单元)
MOV A,R3 ;秒数据放入A(R3为2位十进制数组合)
CLR C ;清进位标志
CJNE A,#60H,ADDMM ;
ADDMM:
JC OUTT0 ;小于60秒时中断退出
LCALL CLR0 ;大于或等于60秒时对秒计时单元清0
MOV R0,#77H ;指向分计时单元(76H-77H)
ACALL DBchange ;
MOV A,R3 ;分数据放入A
CLR C ;清进位标志
CJNE A,#60H,ADDHH ;
ADDHH:
JC OUTT0 ;小于60分时中断退出
ACALL CLR0 ;大于或等于60分时分计时单元清0
MOV R0,#79H ;指向小时计时单元(78H-79H)
ACALL DBchange ;
MOV A,R3 ;时数据放入A
CLR C ;清进位标志
MOV 7BH,#01H
CJNE A,#24H,HOUR ;
HOUR:
JC OUTT0 ;小于24小时中断退出
ACALL CLR0 ;大于或等于24小时小时计时单元清0
OUTT0:
MOV 72H,76H ;中断退出时将分、时计时单元数据移
MOV 73H,77H ;入对应显示单元
MOV 74H,78H ;
MOV 75H,79H ;
POP PSW ;恢复状态字(出栈)
POP ACC ;恢复累加器
SETB ET0 ;开放T0中断
RETI ;中断返回;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 闪动调时 程 序 ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;T1中断服务程序,用作时间调整时调整单元闪烁指示
INTT1:
PUSH ACC ;中断现场保护
PUSH PSW ;
MOV TL1, #0B0H ;装定时器T1定时初值
MOV TH1, #3CH ;
DJNZ R2,INTT1OUT ;0.3秒未到退出中断(50MS中断6次)
MOV R2,#06H ;重装0.3秒定时用初值
CPL 02H ;0.3秒定时到对闪烁标志取反
JB 02H,FLASH1 ;02H位为1时显示单元"熄灭"
MOV 72H,76H ;02H位为0时正常显示
MOV 73H,77H ;
MOV 74H,78H ;
MOV 75H,79H ;
INTT1OUT:
POP PSW ;恢复现场
POP ACC ;
RETI ;中断退出
FLASH1:
JB 01H,FLASH2 ;01H位为1时,转小时熄灭控制
MOV 72H,7AH ;01H位为0时,"熄灭符"数据放入分
MOV 73H,7AH ;显示单元(72H-73H),将不显示分数据
MOV 74H,78H ;
MOV 75H,79H ;
AJMP INTT1OUT ;转中断退出
FLASH2:
MOV 72H,76H ;01H位为1时,"熄灭符"数据放入小时
MOV 73H,77H ;显示单元(74H-75H),小时数据将不显示
MOV 74H,7AH ;
MOV 75H,7AH ;
AJMP INTT1OUT ;转中断退出
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 数据处理程序,把6个数码管显示内容存入相应的地址;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
DBchange:
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 A ;清累加器
MOV @R0,A ;清当前地址单元
DEC R0 ;指向前一地址
MOV @R0,A ;前一地址单元清0
RET ;子程序返回
;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 时钟调整中断程序 ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;当调时按键按下时进入此程序
OUTINTT0:
PUSH ACC;
PUSH PSW;
CLR ET0 ;关定时器T0中断
CLR TR0 ;关闭定时器T0
CLR EX0 ;关闭外部中断0
MOV R2,#06H ;进入调时状态,赋闪烁定时初值
SETB ET1 ;允许T1中断
SETB TR1 ;开启定时器T1
SET2: JNB P1.0,SET1 ;P1.0口为0(键未释放),等待
SETB 00H ;键释放,分调整闪烁标志置1
SET4:
JB P1.0,SET3 ;p1.0没有按下时进入分调整程序
LJMP SETHH ;p1.0按下转调小时状态
SET3:
LCALL DISPLAY ;等待调分按键时时钟显示用
JB P1.1,SET4;等待p1.1按下
LCALL DL1S; MOV R0,#77H;按下时加1分钟操作
LCALL DBchange ;调用加1子程序
MOV A,R3 ;取调整单元数据
CLR C ;清进位标志
CJNE A,#60H,HHH ;调整单元数据与60比较
HHH:
JC SET4 ;调整单元数据小于60转SET4循环
LCALL CLR0 ;调整单元数据大于或等于60时清0
CLR C ;清进位标志
lJMP SET4 SETHH:
CLR 00H ;分闪烁标志清除(进入调小时状态)
SETHH1:
JNB P1.0,SET5 ;等待键释放
SETB 01H ;小时调整标志置1
SET6:
JNB P1.0,SETOUT ;P1.0再次按下时退出时间调整
LCALL DISPLAY;
JB P1.1,SET6;等待按键按下
LCALL DL1S;
MOV R0,#79H ;按下P1.1时加1小时操作
LCALL DBchange ;调加1子程序
MOV A,R3 ;
CLR C ;
CJNE A,#24H,HOUU ;计时单元数据与24比较
HOUU:
JC SET6;小于24转SET6循环
LCALL CLR0 ;大于或等于24时清0操作
AJMP SET6;跳转到SET6循环
SETOUT:
JNB P1.0,SETOUT1 ;调时退出程序。等待键释放
LCALL DISPLAY ;延时削抖
JNB P1.0,SETOUT ;是抖动,返回SETOUT再等待
OUTRETEND:
CLR 01H ;清调小时标志
CLR 00H ;清调分标志
CLR 02H ;清闪烁标志
CLR TR1 ;关闭定时器T1
CLR ET1 ;关定时器T1中断
SETB TR0 ;开启定时器T0
SETB ET0 ;开定时器T0中断(计时开始)
SETB EX0;开启外部中断0
POP PSW;
POP ACC;
RETI ;反回主程序
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
SET1:
LCALL DISPLAY ;键释放等待时调用显示程序(调分)
LJMP SET2 ;防止键按下时无时钟显示
SET5:
LCALL DISPLAY ;键释放等待时调用显示程序(调小时)
LJMP SETHH1 ;防止键按下时无时钟显示
SETOUT1:
LCALL DISPLAY ;退出时钟调整时键释放等待
LJMP SETOUT ;防止键按下时无时钟显示
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 显示程序 ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
; 显示数据在70H-75H单元内,用六位LED共阳数码管显示,P0口输出段码数据,P2口作
; 扫描控制,每个LED数码管亮1MS时间再逐位循环。
DISPLAY:
MOV R1,#70H ;指向显示数据首址
MOV R5,#01H ;扫描控制字初值
PLAY:
MOV A,R5 ;扫描字放入A
MOV P2,A ;从P2口输出
MOV A,@R1 ;取显示数据到A
MOV DPTR,#TAB ;取段码表地址
MOVC A,@A+DPTR ;查显示数据对应段码
MOV P0,A ;段码放入P0口
LCALL DL1MS ;显示1MS
INC R1 ;指向下一地址
MOV A,R5 ;扫描控制字放入A
JB ACC.5,ENDOUT ;ACC.5=1时一次显示结束
RL A ;A中数据循环左移
MOV R5,A ;放回R5内
LJMP PLAY ;跳回PLAY循环
ENDOUT:
MOV P0,#00H ;P0口复位
RET ;子程序返回
TAB: DB 3FH,06H,5BH,4FH,66H,6DH,7DH,07H,7FH,6FH,00H
;共阴段码表 "0""1""2" "3""4""5""6""7" "8""9""不亮"
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;整点报时;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
BELL:
MOV A,7BH;
CLR C;
MOV 7BH,#00H ;
CJNE A,#01H,BELLEND MOV A,79H;
SWAP A;
ORL A,78H;
MOV 7CH ,ABELLRING:
CLR P1.7;
LCALL DL05S;
SETB P1.7;
LCALL DL05S;
DJNZ 7CH,BELLRING; MOV A ,#00H;
MOV 7BH,#00H ;BELLEND:RET
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;省电模式
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
DISCLOSE:
JB P1.3,OUTDISCLOSE;
CLOSE1: MOV P2,#0FFH;
JNB P1.3,CLOSE1;
JB P1.3,CLOSE1;
CLOSE2: JNB P1.3,CLOSE2;
MOV P2,R5;
OUTDISCLOSE:RET;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; 延时程序 ;;
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;
;1MS延时程序,LED显示程序用
DL1MS: MOV R6,#14H
DL1: MOV R7,#19H
DL2: DJNZ R7,DL2
DJNZ R6,DL1
RET
;20MS延时程序,采用调用显示子程序以改善LED的显示闪烁现象
DS20MS: ACALL DISPLAY
ACALL DISPLAY
ACALL DISPLAY
RET
;延时程序,用作按键时间的长短判断
DL1S: LCALL DL05S
LCALL DL05S
RET
DL05S: MOV R3,#20H ;8毫秒*32=0.196秒
DL05S1: LCALL DISPLAY
DJNZ R3,DL05S1
RET
END ;程序结束 但是做这个是自己做的,是可以调时间,不要的话可以把那个部分删除。
③ 如何用51单片机制作简易电子钟
我自己做的一个课程设计,程序运行已经成功了:
汇编程序如下:
ORG0000H
AJMPmain
ORG000BH;定时器0的中断向量地址
AJMPTIME0;跳转到真正的定时器程序处
ORG30H
main:
movdptr,#k1;
start:
MOVP0,#00H;中断预处理程序
MOVP1,#00H
MOVP3,#00H;关所有灯
MOVR0,#00H;软件计数器预清0
MOVR7,#00H
MOVR6,#00H
MOVTMOD,#00000001B;定时/计数器0工作于方式1
MOVTH0,#3CH
MOVTL0,#0B0H;即数15536
SETBEA;开总中断允许
SETBET0;开定时/计数器0允许
SETBTR0;定时/计数器0开始运行
LOOP:;判断时钟进位程序
MOVA,R1
CJNEA,#60,LOOP1;R1等于60吗?
MOVR1,#0
INCR2;R1等于60那么R2加一,R1清0
MOVA,R2
CJNEA,#60,LOOP1;R2等于60吗?
MOVR2,#0
INCR3;R2等于60那么R3加一,R2清0
MOVA,R3
CJNEA,#24,LOOP1;R3等于24吗?
MOVR3,#0
;R3等于24那么R3清0
LOOP1:
mova,r2;如果分钟显示为0,报警
cjnea,#01H,LOOP2
mova,r1
cjnea,#00H,LOOP3
clrp2.0
LOOP3:
mova,r1;如果秒钟显示为1,取消报警
cjnea,#05H,LOOP2
setbp2.0
LOOP2:;显示数据处理
MOVA,R1;取来8位二进制数
MOVB,#100
DIVAB;除以100
MOVA,B;取回余数
MOVB,#10
DIVAB;再除以10
MOVR4,A;商是十位数,存放到R4
MOVR5,B;余数是个位,存放到R5
MOV42H,R4
mOV41H,R5
;R1的BCD码放入41-42H单元
MOVA,R2;取来8位二进制数
MOVB,#100
DIVAB;除以100
MOVA,B;取回余数
MOVB,#10
DIVAB;再除以10
MOVR4,A;商是十位数,存放到R4
MOVR5,B;余数是个位,存放到R5
MOV44H,R4
MOV43H,R5
;R2的BCD码放入43-44H单元
MOVA,R3;取来8位二进制数
MOVB,#100
DIVAB;除以100
MOVA,B;取回余数
MOVB,#10
DIVAB;再除以10
MOVR4,A;商是十位数,存放到R4
MOVR5,B;余数是个位,存放到R5
MOV46H,R4
mov45H,R5
;R3的BCD码放入45-46H单元
L1:
mova,41h
movca,@a+dptr
mov51h,a
mova,42h
movca,@a+dptr
mov52h,a
mova,43h
movca,@a+dptr
mov53h,a
mova,44h
movca,@a+dptr
mov54h,a
mova,45h
movca,@a+dptr
mov55h,a
mova,46h
movca,@a+dptr
mov56h,a;将显示代码送入51H--56H
display:;显示程序
movp0,#0ffH;一
movp0,51H
movP1,#0feH
acalldelay1S
movp1,#0ffH
movp0,#0ffH;二
movp0,52h
movP1,#0fdH
acalldelay1S
movp1,#0ffH
movp0,#0ffH;三
movp0,53h
movP1,#0fbH
acalldelay1S
movp1,#0ffH
movp0,#0ffH;四
movp0,54h
movP1,#0f7H
acalldelay1S
movp1,#0ffH
movp0,#0ffH;五
movp0,55h
movP1,#0efH
acalldelay1S
movp1,#0ffH
movp0,#0ffH;六
movp0,56h
movP1,#0dfH
acalldelay1S
movp1,#0ffH
;将51H-56H数据显示,
AJMPLOOP;工作处理程序
TIME0:;定时器0的中断处理程序
INCR7
MOVA,R7
CJNEA,#20,T_RET;R7单元中的值到了20了吗?
T_L1:
INCR1;到了20那么R1+1
MOVR7,#0;清软件计数器
T_RET:;二十分之一秒时的程序
incr6
mova,r6
cjner6,#3,T_over;调节时间的延迟时间
movr6,#0
jbp2.1,T_over1
mova,r2
adda,#1
cjnea,#60,T_tiao1
mova,#0
T_tiao1:
movr2,a
T_over1:
jbp2.2,T_over
mova,r3
adda,#1
cjnea,#24,T_tiao2
mova,#0
T_tiao2:
movr3,a
T_over:
MOVTH0,#3CH
MOVTL0,#0B0H;如果没到一秒重置定时常数
reti
delay1s:MOV31H,#100
D1:MOV32H,#30
D2:DJNZ32H,D2
DJNZ31H,D1
RET
delay2s:MOV33H,#255
D3:MOV34H,#255
D4:DJNZ34H,D4
DJNZ33H,D3
RET
K1:db3fh,06h,5bh,4fh,66h,6dh,7dh,07h,7fh,6fh
④ 单片机电子时钟
PCF8583的电路图和程序我这里有,可以HI我
加上数码管和键盘就可以用了。
⑤ 51单片机电子时钟
才1毛钱
⑥ 51单片机电子钟
看看这个简单的仿真实例,供参考
⑦ 求c51单片机电子钟程序(c语言)
#include <reg51.h>
#define uchar unsigned char //定义unsigned int为uint
#define uint unsigned int //定义unsigned uchar为uchar
sbit LCD_RS = P2^0 ;
sbit LCD_RW = P2^1 ;
sbit LCD_EN = P2^2 ;
sbit D_SDA = P2^6; //定义74HC164数据线为P2.6端口
sbit D_SCL = P2^7; //定义74HC164数据线为P2.7端口
sbit CLK = P1^3; /*实时时钟时钟线引脚 */
sbit IO = P1^4; /*实时时钟数据线引脚 */
sbit RST = P1^5; /*实时时钟复位线引脚 */
sbit ACC0 = ACC^0;
sbit ACC7 = ACC^7;
uchar time[8] = {0x50,0x30,0x19,0x30,0x12,0x06,0x06};
//========= 延时函数 ============
//延时时间以1ms为单位
//s决定延时时间长短
void delay_ms(uint s)
{
uint x;
for(s;s>0;s--)
{
x = 200;
while(x--);
}
}
//========= 送出一个字节给74HC164(实现串并转换) ==========
void send_out(unsigned char out)//传送一个字节8位
{
uchar i;
D_SCL = 0;
for (i=8;i>=1;i--)
{
D_SDA = out&0x80; //送数据到数据口
D_SCL = 1; //时钟线置1
D_SCL = 0; //送一时钟
out<<=1; //左移
}
}
//========= 写命令函数 ==========
void lcd_wcmd(uchar cmd)
{
LCD_RS = 0;
LCD_RW = 0;
LCD_EN = 0;
send_out(cmd);
LCD_EN = 1;
LCD_EN = 0 ;
}
//========= 写数据函数 ==========
void lcd_wdat(uchar dat)
{
LCD_RS = 1;
LCD_RW = 0;
LCD_EN = 0;
send_out(dat);
LCD_EN = 1;
LCD_EN = 0;
}
//========= LCD初始化函数 ==========
void lcd_init()
{
lcd_wcmd(0x38);
delay_ms(1);
lcd_wcmd(0x0c); //显示开,关光标
delay_ms(1);
lcd_wcmd(0x06); //向右移动光标
delay_ms(1);
lcd_wcmd(0x01); //清除LCD显示屏
delay_ms(1);
}
//========== 往DS1302写入1Byte数据 (内部函数) =============
void w_byte(uchar dat)
{
uchar i;
for(i=8; i>0; i--)
{
IO = dat & 0x01;
CLK = 1;
CLK = 0;
dat = dat >> 1;
}
}
//======== 从DS1302读取1Byte数据 (内部函数) ===================
uchar r_byte(void)
{
uchar i;
for(i=8; i>0; i--)
{
ACC = ACC >> 1;
ACC7 = IO;
CLK = 1;
CLK = 0;
}
return(ACC);
}
//========== 指定地址往DS1302写入1Byte数据 (内部函数) =============
void write_byte(uchar addr, uchar dat)
{
RST = 0;
CLK = 0;
RST = 1;
w_byte(addr);
w_byte(dat);
CLK = 1;
RST = 0;
}
//========== 指定地址往DS1302读1Byte数据 (内部函数) =============
uchar read_byte(uchar addr)
{
uchar ucData;
RST = 0;
CLK = 0;
RST = 1;
w_byte(addr);
ucData = r_byte();
CLK = 1;
RST = 0;
return(ucData);
}
//============ 设置ds1302日期和时间 =============
void write_ds1302(uchar *p)
{
uchar i;
uchar addr = 0x80;
write_byte(0x8e,0x00); // 控制命令,WP=0,写操作
for(i =7; i>0; i--)
{
write_byte(addr,*p); // 秒 分 时 日 月 星期 年
p++;
addr +=2;
}
write_byte(0x8e,0x80); // 控制命令,WP=1,写保护
}
//============ 读ds1302当前日期和时间 =============
void read_ds1302(uchar *p)
{
uchar i;
uchar addr = 0x81;
for (i=0; i<7; i++)
{
*p = read_byte(addr); //格式为: 秒 分 时 日 月 星期 年
addr += 2;
p++;
}
}
//============ 显示函数 ===================
void lcd_disp()
{
uchar addr = 4;
lcd_wcmd(0x80 + addr);
lcd_wdat(((time[2]>>4)&0x0f)+0x30); //显示小时
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat((time[2]&0x0f)+0x30);
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat(':'); //显示":"
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat(((time[1]>>4)&0x0f)+0x30); //显示分
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat((time[1]&0x0f)+0x30);
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat(':'); //显示":"
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat(((time[0]>>4)&0x0f)+0x30); //显示秒
addr++;
lcd_wcmd(0x80 + addr);
lcd_wdat((time[0]&0x0f)+0x30);
addr = 2;
lcd_wcmd(0xc0 + addr); //在第二行显示年月日和星期
lcd_wdat('2'); //显示2
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat('0'); //显示0
addr++;
lcd_wdat(((time[6]>>4)&0x0f)+0x30); //年
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat((time[6]&0x0f)+0x30);
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat(' ');
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat(((time[4]>>4)&0x0f)+0x30); //显示月
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat((time[4]&0x0f)+0x30);
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat(' ');
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat(((time[3]>>4)&0x0f)+0x30); //显示日
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat((time[3]&0x0f)+0x30);
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat(' ');
addr++;
lcd_wcmd(0xc0 + addr);
lcd_wdat((time[5]&0x0f)+0x30); //显示星期
}
//=========== 主函数 ===============
void main()
{
lcd_init(); // 初始化LCD
write_ds1302(time);
while(1)
{
read_ds1302(time); //读DS1302数据
lcd_disp(); //LCD显示
delay_ms(500); //延时0.5秒
}
}
这是一个电子时钟,在LCD1602上显示,时钟芯片是DS1302
⑧ 基于单片机的电子时钟
首先,硬件的选取:
1,单片机,其控制系统不算复杂,使用简单的8位机即可,然后就是时钟的选择,本设计基于其实时性要求不高,所以时钟尽量选择低一点,可以用3M、6M的时钟,因为时钟选择越高其单片机的功耗就越大,甚至出现超频现象(不过现在的单片机时钟一般都能达到50M了)!
2,显示部分,根据本例程的要求,可以选用6个8位数码管进行显示。
3,驱动电路,数码管驱动芯片有很多,常见的有串行通信的如74LS164、74HC595、SM1617等常用LED串行驱动芯片。
4,时钟芯片的选择,单片机上面有定时器,但是实现长时间的定时必定不准确,误差很大,还得经常对时,所以建议你选用一款时钟芯片,常用的DS1302等。
5,用户界面,你就用几个按键实现用户操作就行了。
软件:首先显示部分的变成,在网上有很多例程,你只需要简单的移植就可以使用了,时钟芯片DS1302典型的IIC总线驱动模式,你可以使用单片机中IIC外设来访问,按键部分使用扫描和外部中断都可!
自己动手丰衣足食,写了这么多,希望对你有用,加油
⑨ 51单片机实现电子时钟
可以用中断或者专门给调试这个子程序,调试时执行子程序,调试完跳出