一、系统方案
1、本设计采用STC15单片机作为主控器。
2、DS18B20采集温度值送到液晶1602显示。
3、DS1302计时,日期送到液晶1602显示。
4、LM35采集另一路温度值送到数码管显示。
二、硬件设计
原理图如下:
三、单片机软件设计
1、首先是系统初始化
/IO初始化为准双向 /
void IO_Init()
{
P0M0 = 0x00;P0M1 = 0x00;
P1M0 = 0x00;P1M1 = 0x00;
P2M0 = 0x00;P2M1 = 0x00;
P3M0 = 0x00;P3M1 = 0x00;
P4M0 = 0xFF;P4M1 = 0x00;
P5M0 = 0x00;P5M1 = 0x00;
P6M0 = 0x00;P6M1 = 0x00;
P7M0 = 0x00;P7M1 = 0x00;
P0 = 0x00;P1 = 0x00;P2 = 0x00;P3 = 0x00;
P4 = 0x00;P5 = 0x00;P6 = 0x00;P7 = 0x00;
2、液晶显示程序
//写指令函数=
void LCD_write_command(uchar command)
{
LCD_DB=command;
LCD_RS=0;//指令
LCD_RW=0;//写入
LCD_E=1;
LCD_E=0;
delay_ms(2);//等待执行完毕 。。。
}
//=写数据函数=
void LCD_write_data(uchar dat)
{ LCD_DB=dat;
LCD_RS=1;//数据寄存器
LCD_RW=0;//写入数据
LCD_E=1;
LCD_E=0;
delay_ms(2);//等待程序执行完毕 。。。
}
//=初始化函数=
void LCD_init(void)
{
LCD_write_command(0x38);//设置8位格式,2行,57
LCD_write_command(0x38);//设置8位格式,2行,5 7
LCD_write_command(0x0c);//整体显示,关光标,不闪烁
LCD_write_command(0x06);//设定输入方式,增量不移位
LCD_write_command(0x01);//清屏显示
LCD_write_command(0x80);
delay_ms(2);
}
//光标定位===========
void gotoxy(uchar x,uchar y)
{
uchar address;
if(y1)
address=0x80+x;
else
address=0xc0+x;
LCD_write_command(address);
}
//=显示一个字符串========
void LCD_display(uchar *dat)
{
while(*dat!='\0')
{ LCD_write_data(*dat);
dat++;
}
}
3、DS1302程序
#include"ds1302.h"
//---DS1302写入和读取时分秒的地址命令---//
//---秒分时日月周年 最低位读写位;-------//
uchar code READ_RTC_ADDR[7] = {0x81, 0x83, 0x85, 0x87, 0x89, 0x8b, 0x8d};
uchar code WRITE_RTC_ADDR[7] = {0x80, 0x82, 0x84, 0x86, 0x88, 0x8a, 0x8c};
//---DS1302时钟初始化2022年5月23日星期一17点29分00秒。---//
//---存储顺序是秒分时日月周年,存储格式是用BCD码---//
uchar time[7] = {0x00, 0x30, 0x17, 0x01, 0x06, 0x01, 0x22};
/*******************************************************************************
- 函 数 名 : Ds1302Write
- 函数功能 : 向DS1302命令(地址+数据)
- 输 入 : addr,dat
- 输 出 : 无
*******************************************************************************/
void Ds1302Write(uchar addr, uchar dat)
{
uchar n;
RST = 0;
nop();
SCLK = 0;//先将SCLK置低电平。
_nop_();
RST = 1; //然后将RST(CE)置高电平。
_nop_();
for (n=0; n<8; n++)//开始传送八位地址命令
{
DSIO = addr & 0x01;//数据从低位开始传送
addr >>= 1;
SCLK = 1;//数据在上升沿时,DS1302读取数据
_nop_();
SCLK = 0;
_nop_();
}
for (n=0; n<8; n++)//写入8位数据
{
DSIO = dat & 0x01;
dat >>= 1;
SCLK = 1;//数据在上升沿时,DS1302读取数据
_nop_();
SCLK = 0;
_nop_();
}
RST = 0;//传送数据结束
_nop_();
}
/*******************************************************************************
- 函 数 名 : Ds1302Read
- 函数功能 : 读取一个地址的数据
- 输 入 : addr
- 输 出 : dat
*******************************************************************************/
uchar Ds1302Read(uchar addr)
{
uchar n,dat,dat1;
RST = 0;
nop();
SCLK = 0;//先将SCLK置低电平。
_nop_();
RST = 1;//然后将RST(CE)置高电平。
_nop_();
for(n=0; n<8; n++)//开始传送八位地址命令
{
DSIO = addr & 0x01;//数据从低位开始传送
addr >>= 1;
SCLK = 1;//数据在上升沿时,DS1302读取数据
_nop_();
SCLK = 0;//DS1302下降沿时,放置数据
_nop_();
}
_nop_();
for(n=0; n<8; n++)//读取8位数据
{
dat1 = DSIO;//从最低位开始接收
dat = (dat>>1) | (dat1<<7);
SCLK = 1;
_nop_();
SCLK = 0;//DS1302下降沿时,放置数据
_nop_();
}
RST = 0;_nop_(); //以下为DS1302复位的稳定时间,必须的。
SCLK = 1;_nop_();
DSIO = 0;_nop_();
DSIO = 1;_nop_();
return dat;
}
/*******************************************************************************
- 函 数 名 : Ds1302Init
- 函数功能 : 初始化DS1302.
- 输 入 : 无
- 输 出 : 无
*******************************************************************************/
void Ds1302Init()
{
uchar n;
Ds1302Write(0x8E,0X00); //禁止写保护,就是关闭写保护功能
for (n=0; n<7; n++)//写入7个字节的时钟信号:分秒时日月周年
{
Ds1302Write(WRITE_RTC_ADDR[n],time[n]);
}
Ds1302Write(0x8E,0x80); //打开写保护功能
}
/*******************************************************************************
- 函 数 名 : Ds1302ReadTime
- 函数功能 : 读取时钟信息
- 输 入 : 无
- 输 出 : 无
*******************************************************************************/
void Ds1302ReadTime()
{
uchar n;
for (n=0; n<7; n++)//读取7个字节的时钟信号:分秒时日月周年
{
time[n] = Ds1302Read(READ_RTC_ADDR[n]);
}
}
4、核心算法程序
void main()
{
Init_DS18B20();
LCD_init();
judge=Init_DS18B20();
Ds1302Init();
gotoxy(0,1);
LCD_display("20204397");
InitADC();
while(1)
{
Ds1302ReadTime();
nyrstr[0] = 2 +'0'; //"年"
nyrstr[1] = 0 +'0';
nyrstr[2] = (time[6] >> 4) +'0'; //"年"
nyrstr[3] = (time[6]&0x0F) +'0';
nyrstr[4] = '-';
nyrstr[5] = (time[4] >> 4) +'0'; //"月"
nyrstr[6] = (time[4]&0x0F) +'0';
nyrstr[7] = '-';
nyrstr[8] = (time[3] >> 4) +'0'; //"日"
nyrstr[9] = (time[3]&0x0F) +'4';
TempPhoto =GetADCResultint(1); //实时读取P1.1通道的AD转换结果
TempPhoto=TempPhoto5.020/1023;
//Disp_Temperature(TempPhoto);
gotoxy(0,1);
LCD_display(nyrstr);
display();
LCD_write_command(0xc0);
wendu=ReadTemperature();
Disp_Temperature(wendu);
}
}
四、 proteus仿真设计
Proteus软件是一款应用比较广泛的工具,它可以在没有硬件平台的基础上通过自身的软件仿真出硬件平台的运行情况,这样就可以通过软件仿真来验证我们设计的方案有没有问题,如果有问题,可以重新选择器件,连接器件,直到达到我们设定的目的,避免我们搭建实物的时候,如果当初选择的方案有问题,我们器件都已经焊接好了,再去卸载下去,再去焊接新的方案的器件,测试,这样会浪费人力和物力,也给开发者带来一定困惑,Proteus仿真软件就很好的解决这个问题,我们在设计之初,就使用该软件进行模拟仿真,测试,选择满足我们设计的最优方案。最后根据测试没问题的仿真图纸,焊接实物,调试,最终完成本设计的作品。