这是我写的一个成功的,你仔细对比下就知道了!
#include<reg52h>
#include<intrinsh>
#define U8 unsigned char
#define U16 unsigned int
#define BOOL bit
U8 secondh,secondl,minuteh,minutel,hourh,hourl,second,minute,hour;
U8 monthl,monthh,month,day,datel,dateh,date,year,yearl,yearh;
/定义LCD功能管脚/
sbit LCDRS = P2^6 ; //控制LCD数据或命令的读写
sbit LCDRW = P2^5 ; //控制LCD读写信号
sbit LCDEN = P2^7 ; //使能LCD
//
/定义DS1302功能管脚/
sbit DSRST = P3^5 ; //DS1302复位管脚
sbit DSIO = P3^4 ; //控制DS1302数据传输
sbit DSCLK = P3^6 ; //DS1302时钟
//
/延时函数/
void delay(U16 time)
{
U16 time1,time2;
for(time1=time;time1>0;time1--)
for(time2=110;time2>0;time2--);
}
//
/检测LCD忙碌状态/
BOOL judge_lcd()
{
BOOL flag;
LCDEN = 0;
LCDRS = 0;
LCDRW = 1;
LCDEN = 1;
delay(1);
flag=(BOOL)(P0&0x80); //状态位第7位为1时禁止读写,为0时可读写
LCDEN = 0;
return flag;
}
//
/实现LCD写指令/
void write_lcd_command(U16 command)
{
while(judge_lcd()); //检测LCD是否为忙碌状态
P0 = command ;
LCDRS = 0 ;
LCDRW = 0 ;
LCDEN = 1 ;
delay(1);
LCDEN = 0 ;
}
//
/实现LCD写数据/
void write_lcd_date(U16 date)
{
while(judge_lcd()); //检测LCD是否为忙碌状态
P0 = date ;
LCDRS = 1 ;
LCDRW = 0 ;
LCDEN = 1 ;
delay(1);
LCDEN = 0 ;
}
//
/实现LCD写指令及数据/
void write_lcd_com_date(U16 addr,U16 date)
{
write_lcd_command(addr);
delay(1);
write_lcd_date(date);
}
//
/初始化LCD/
void init_lcd()
{
write_lcd_command(0x38); //设置16x2显示,5x7点阵,8位数据口
delay(1);
write_lcd_command(0x0c); //开显示,不显示光标,光标不闪烁
delay(1);
write_lcd_command(0x06); //读写一个字符后,地址自动加1,不移动屏幕
delay(1);
write_lcd_command(0x01); //清屏
delay(1);
}
//
/读DS1302/
U8 read_ds1302(U8 addr)
{
U8 i,temp=0x00;
DSRST=0;
DSIO=1;
DSCLK=0;
DSRST=1;
for (i=0;i<8;i++) //循环8次 写入地址数据
{
DSCLK=0;
DSIO=addr&0x01; //每次传输低字节
addr>>=1; //右移一位
DSCLK=1;
}
delay(1);
if(DSIO)
temp|=0x80; //每次传输低字节
DSCLK=0;
temp>>=1;
for (i=0;i<7;i++) //循环8次 读取数据
{
DSCLK=0;
if(DSIO)
temp|=0x80; //每次传输低字节
DSCLK=1;
temp>>=1; //右移一位
}
DSCLK=1;
DSRST=0;
DSIO=0;
return temp; //返回
}
//
/写DS1302/
void write_ds1302(U8 addr, U8 date)
{
U8 i;
DSRST=0;
DSCLK=0;
DSRST=1;
for (i=0;i<8;i++) //循环8次 写入地址数据
{
DSCLK=0;
_nop_();
DSIO=addr&0x01; //每次传输低字节
addr>>=1; //右移一位
DSCLK=1;
_nop_();
}
for (i=0;i<8;i++) //循环8次 写入数据
{
DSCLK=0;
_nop_();
DSIO=date&0x01; //每次传输低字节
date>>=1; //右移一位 DSCLK=1
DSCLK=1;
_nop_();
}
DSRST=0;
delay(1);
}
//
/初始化DS1302/
void init_ds1302()
{
//读秒
if((((second=read_ds1302(0x81))>>7)&&0x01)==1)
{
write_ds1302(0x8e,0x00); //关闭写保护
write_ds1302(0x80,0x00); //00秒
write_ds1302(0x82,0x54); //54分
write_ds1302(0x84,0x17); //17点
write_ds1302(0x86,0x14); //14日
write_ds1302(0x88,0x10); //10月
write_ds1302(0x8a,0x05); //星期五
write_ds1302(0x8c,0x11); //2011年
write_ds1302(0x8e,0x80); //允许写保护
}
}
//
/主函数/
void main()
{
init_lcd();
delay(2);
write_lcd_com_date(0x81,'2');
write_lcd_com_date(0x82,'0');
delay(1);
init_ds1302();
delay(1);
while(1)
{
second=read_ds1302(0x81);
secondl=second&0x0f;
secondh=second>>4;
minute=read_ds1302(0x83);
minutel=minute&0x0f;
minuteh=minute>>4;
hour=read_ds1302(0x85);
hourl=hour&0x0f;
hourh=hour>>4;
month=read_ds1302(0x89);
monthl=month&0x0f;
monthh=month>>4;
year=read_ds1302(0x8d);
yearl=year&0x0f;
yearh=year>>4;
date=read_ds1302(0x87);
datel=date&0x0f;
dateh=date>>4;
day=read_ds1302(0x8b);
day=day&0x0f;
delay(15);
write_lcd_com_date(0xca,0x30+secondl);
write_lcd_com_date(0xc9,0x30+secondh);
write_lcd_com_date(0xc8,':');
write_lcd_com_date(0xc7,0x30+minutel);
write_lcd_com_date(0xc6,0x30+minuteh);
write_lcd_com_date(0xc5,':');
write_lcd_com_date(0xc4,0x30+hourl);
write_lcd_com_date(0xc3,0x30+hourh);
write_lcd_com_date(0x84,0x30+yearl);
write_lcd_com_date(0x83,0x30+yearh);
write_lcd_com_date(0x85,'-');
write_lcd_com_date(0x87,0x30+monthl);
write_lcd_com_date(0x86,0x30+monthh);
write_lcd_com_date(0x88,'-');
write_lcd_com_date(0x8a,0x30+datel);
write_lcd_com_date(0x89,0x30+dateh);
write_lcd_com_date(0x8c,0x30+day);
}
}
就是十进制和十六进制的转换
写的时候十进制转换成十六进制
十进制 34
0x34 = (34/10)16 + 34%10;
读的时候十六进制转换成十进制
十六进制 0x34
34 = (0x34/16)10 + 0x34%16;
CE和SCLK是DS1302芯片的信号线,但也是与单片机的PIO口连接的;
因此初始化CE和SCLK,也就是初始化单片机的PIO口了;
在代码中,你可以去看看 DS1302_CE、DS1302_SCLK 的定义就明白了;
DS1302程序:
#include "ds1302h"
unsigned char time_buf1[8] = {20,9,3,13,18,51,00,6};//空年月日时分秒周
unsigned char time_buf[8] ; //空年月日时分秒周
/------------------------------------------------
向DS1302写入一字节数据
------------------------------------------------/
void Ds1302_Write_Byte(unsigned char addr, unsigned char d)
{
unsigned char i;
RST_SET;
//写入目标地址:addr
addr = addr & 0xFE; //最低位置零
for (i = 0; i < 8; i ++)
{
if (addr & 0x01)
{
IO_SET;
}
else
{
IO_CLR;
}
SCK_SET;
SCK_CLR;
addr = addr >> 1;
}
//写入数据:d
for (i = 0; i < 8; i ++)
{
if (d & 0x01)
{
IO_SET;
}
else
{
IO_CLR;
}
SCK_SET;
SCK_CLR;
d = d >> 1;
}
RST_CLR; //停止DS1302总线
}
/------------------------------------------------
从DS1302读出一字节数据
------------------------------------------------/
unsigned char Ds1302_Read_Byte(unsigned char addr)
{
unsigned char i;
unsigned char temp;
RST_SET;
//写入目标地址:addr
addr = addr | 0x01;//最低位置高
for (i = 0; i < 8; i ++)
{
if (addr & 0x01)
{
IO_SET;
}
else
{
IO_CLR;
}
SCK_SET;
SCK_CLR;
addr = addr >> 1;
}
//输出数据:temp
for (i = 0; i < 8; i ++)
{
temp = temp >> 1;
if (IO_R)
{
temp |= 0x80;
}
else
{
temp &= 0x7F;
}
SCK_SET;
SCK_CLR;
}
RST_CLR; //停止DS1302总线
return temp;
}
/------------------------------------------------
向DS1302写入时钟数据
------------------------------------------------/
void Ds1302_Write_Time(void)
{
unsigned char i,tmp;
for(i=0;i<8;i++)
{ //BCD处理
tmp=time_buf1[i]/10;
time_buf[i]=time_buf1[i]%10;
time_buf[i]=time_buf[i]+tmp16;
}
Ds1302_Write_Byte(ds1302_control_add,0x00); //关闭写保护
Ds1302_Write_Byte(ds1302_sec_add,0x80); //暂停
//Ds1302_Write_Byte(ds1302_charger_add,0xa9); //涓流充电
Ds1302_Write_Byte(ds1302_year_add,time_buf[1]); //年
Ds1302_Write_Byte(ds1302_month_add,time_buf[2]); //月
Ds1302_Write_Byte(ds1302_date_add,time_buf[3]); //日
Ds1302_Write_Byte(ds1302_day_add,time_buf[7]); //周
Ds1302_Write_Byte(ds1302_hr_add,time_buf[4]); //时
Ds1302_Write_Byte(ds1302_min_add,time_buf[5]); //分
Ds1302_Write_Byte(ds1302_sec_add,time_buf[6]); //秒
Ds1302_Write_Byte(ds1302_day_add,time_buf[7]); //周
Ds1302_Write_Byte(ds1302_control_add,0x80); //打开写保护
}
/------------------------------------------------
从DS1302读出时钟数据
------------------------------------------------/
void Ds1302_Read_Time(void)
{
unsigned char i,tmp;
time_buf[1]=Ds1302_Read_Byte(ds1302_year_add); //年
time_buf[2]=Ds1302_Read_Byte(ds1302_month_add); //月
time_buf[3]=Ds1302_Read_Byte(ds1302_date_add); //日
time_buf[4]=Ds1302_Read_Byte(ds1302_hr_add); //时
time_buf[5]=Ds1302_Read_Byte(ds1302_min_add); //分
time_buf[6]=(Ds1302_Read_Byte(ds1302_sec_add))&0x7F;//秒
time_buf[7]=Ds1302_Read_Byte(ds1302_day_add); //周
for(i=0;i<8;i++)
{ //BCD处理
tmp=time_buf[i]/16;
time_buf1[i]=time_buf[i]%16;
time_buf1[i]=time_buf1[i]+tmp10;
}
}
/------------------------------------------------
DS1302初始化
------------------------------------------------/
void Ds1302_Init(void)
{
RST_CLR; //RST脚置低
SCK_CLR; //SCK脚置低
Ds1302_Write_Byte(ds1302_sec_add,0x00);
}
主程序MAIN:
/-----------------------------------------------
名称:DS1302时钟数码管显示
论坛:>
编写:shifang
日期:20095
修改:无
内容:DS1302实时时钟数码管显示,只显示时间。并通过4个按键加减小时、分钟,其他参数调节自行添加
------------------------------------------------/
#include<reg52h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义
#include "ds1302h"
#define KeyPort P3 //定义按键端口
#define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换
sbit LATCH1=P2^2;//定义锁存使能端口 段锁存
sbit LATCH2=P2^3;// 位锁存
bit ReadTimeFlag;//定义读时间标志
unsigned char code dofly_DuanMa[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 显示段码值0~9
unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码
unsigned char TempData[8]; //存储显示值的全局变量
void DelayUs2x(unsigned char t);//us级延时函数声明
void DelayMs(unsigned char t); //ms级延时
void Display(unsigned char FirstBit,unsigned char Num);//数码管显示函数
unsigned char KeyScan(void);//键盘扫描
void Init_Timer0(void);//定时器初始化
/------------------------------------------------
主函数
------------------------------------------------/
void main (void)
{
unsigned char num;
Init_Timer0();
Ds1302_Init();
while (1) //主循环
{
num=KeyScan();
switch(num)
{
case 1:time_buf1[4]++;if(time_buf1[4]==24)time_buf1[4]=0;
Ds1302_Write_Time();break; //正常时间 小时 加1
case 2:time_buf1[4]--;if(time_buf1[4]==255)time_buf1[4]=23;
Ds1302_Write_Time();break; //正常时间 小时减1
case 3:time_buf1[5]++;if(time_buf1[5]==60)time_buf1[5]=0;
Ds1302_Write_Time();break;//分加1
case 4:time_buf1[5]--;if(time_buf1[5]==255)time_buf1[5]=59;
Ds1302_Write_Time();break; //分减1
default:break;
}
if(ReadTimeFlag==1)
{
ReadTimeFlag=0;
Ds1302_Read_Time();
//数据的转换,因我们采用数码管0~9的显示,将数据分开
TempData[0]=dofly_DuanMa[time_buf1[4]/10]; //时
TempData[1]=dofly_DuanMa[time_buf1[4]%10];
TempData[2]=0x40; //加入"-"
TempData[3]=dofly_DuanMa[time_buf1[5]/10]; //分
TempData[4]=dofly_DuanMa[time_buf1[5]%10];
TempData[5]=0x40;
TempData[6]=dofly_DuanMa[time_buf1[6]/10]; //秒
TempData[7]=dofly_DuanMa[time_buf1[6]%10];
}
}
}
/------------------------------------------------
uS延时函数,含有输入参数 unsigned char t,无返回值
unsigned char 是定义无符号字符变量,其值的范围是
0~255 这里使用晶振12M,精确延时请使用汇编,大致延时
长度如下 T=tx2+5 uS
------------------------------------------------/
void DelayUs2x(unsigned char t)
{
while(--t);
}
/------------------------------------------------
mS延时函数,含有输入参数 unsigned char t,无返回值
unsigned char 是定义无符号字符变量,其值的范围是
0~255 这里使用晶振12M,精确延时请使用汇编
------------------------------------------------/
void DelayMs(unsigned char t)
{
while(t--)
{
//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);
}
}
/------------------------------------------------
显示函数,用于动态扫描数码管
输入参数 FirstBit 表示需要显示的第一位,如赋值2表示从第三个数码管开始显示
如输入0表示从第一个显示。
Num表示需要显示的位数,如需要显示99两位数值则该值输入2
------------------------------------------------/
void Display(unsigned char FirstBit,unsigned char Num)
{
static unsigned char i=0;
DataPort=0; //清空数据,防止有交替重影
LATCH1=1; //段锁存
LATCH1=0;
DataPort=dofly_WeiMa[i+FirstBit]; //取位码
LATCH2=1; //位锁存
LATCH2=0;
DataPort=TempData[i]; //取显示数据,段码
LATCH1=1; //段锁存
LATCH1=0;
i++;
if(i==Num)
i=0;
}
/------------------------------------------------
定时器初始化子程序
------------------------------------------------/
void Init_Timer0(void)
{
TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
//TH0=0x00; //给定初值
//TL0=0x00;
EA=1; //总中断打开
ET0=1; //定时器中断打开
TR0=1; //定时器开关打开
}
/------------------------------------------------
定时器中断子程序
------------------------------------------------/
void Timer0_isr(void) interrupt 1
{
static unsigned int num;
TH0=(65536-2000)/256; //重新赋值 2ms
TL0=(65536-2000)%256;
Display(0,8); // 调用数码管扫描
num++;
if(num==50) //大致100ms
{
num=0;
ReadTimeFlag=1; //读标志位置1
}
}
/------------------------------------------------
按键扫描函数,返回扫描键值
------------------------------------------------/
unsigned char KeyScan(void)
{
unsigned char keyvalue;
if(KeyPort!=0xff)
{
DelayMs(10);
if(KeyPort!=0xff)
{
keyvalue=KeyPort;
while(KeyPort!=0xff);
switch(keyvalue)
{
case 0xfe:return 1;break;
case 0xfd:return 2;break;
case 0xfb:return 3;break;
case 0xf7:return 4;break;
case 0xef:return 5;break;
case 0xdf:return 6;break;
case 0xbf:return 7;break;
case 0x7f:return 8;break;
default:return 0;break;
}
}
}
return 0;
}
以上就是关于单片机1602液晶显示的DS1302实时时钟帮我分析以下程序运行步骤,详细解释一下,尤其是Bcd转换全部的内容,包括:单片机1602液晶显示的DS1302实时时钟帮我分析以下程序运行步骤,详细解释一下,尤其是Bcd转换、51单片机DS1302时钟芯片BCD码转换问题、51单片机中DS1302芯片的引脚配置原理是什么等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)