基于51单片机的的红外测温计并显示实时时间
前言
了解有关智能仪器设计所必需的基础知识;掌握智能仪器设计的方案论证、硬件/软件设计制作和仿真调试的一般方法和过程;提高学生综合应用专业技术的能力;培养学生具有实际动手进行设计和研制开发智能仪器的基本能力。
提示:以下是本篇文章正文内容,下面案例可供参考
一、功能实现
实时时钟显示(通过DS1302),实现红外测温显示(通过MLX90614)并实现对所测温度的保存以及记录查看(具体实现为5组)。输出显示采用LCD1602或LED七段数码显示器。
二、设计原理
2.1 时钟模块
DS1302是美国DALLAS公司推出的I2C总线接口实时时钟芯片,它可独立于CPU 工作,不受CPU主晶振及其电容的影响,且计时准确,月累积误差一般小于10秒。芯片还具有主电源掉电情况下的时钟保护电路,DS1302的时钟靠后备电池维持工作,拒绝CPU对其读出和写入访问。同时还具有备用电源自动切换控制电路,因而可在主电源掉电和其它一些恶劣环境场合中保证系统时钟的定时准确性。
DS1302具有产生秒、分、时、日、月、年等功能,且具有闰年自动调整功能。同时,DS1302芯片内部还集成有一定容量、具有掉电保护特性的静态RAM,可用于保存一些关键数据。
2.2 显示模块
通过LCD1602作为显示设备,要求开机时显示屏分两行显示,第一行显示日期(年-月-日)和星期,第二行显示时间(时:分:秒)。持续一秒之后第一行显示时间(时:分:秒),第二行显示测得的温度。
2.3 按键模块
按键用来作为模式选择。要求至少设计4个按键,按键1作为模式选择键(按下时进入查看温度记录模块),按键2、按键3分别为下一个、上一个(实现对保存的记录的查看,总共会保存5组记录),在按键4按下时,退出查看温度记录模式,返回测温以及显示模式。仿真中采用4*4矩阵键盘,只有第一行按键有效,其余按键作为保留,可为以后的功能扩展备用。
2.4 红外测温模块
MLX90614是一款由Melexis研发并生产的测温器,可适用于高精度无接触远程测温、汽车空调系统、工业过程温度控制、医疗、家电、人体测温等多种领域。其内部包括红外热电堆感应器MLX81101和专为适用于这款传感器输出而设计的信号处理芯片MLX90302。MLX90302在信号调节芯片中使用了先进的低噪音放大器,一枚17bitADC以及功能强大的DSP元件,从而实现高精确度温度测量,计算并储存于RAM中的环境温度以及物体温度可实现0.01℃的解析度的数据,并且可以通过双线标准SMBus输出获得0.02℃解析度或者通过10bitPWM输出获得。
三、软件设计
#include <reg52.h>
#include <intrins.h>
#include <string.h>
#include <stdio.h>
#define uint unsigned int
#define uchar unsigned char
#define Nack_number 10
//************************** mlx90614 ***********************************
//command mode 命令模式
//#define RamAccess 0x00 //对 RAM 操作
//#define EepomAccess 0x20 //对 EEPRAM 操作
//#define Mode 0x60 //进入命令模式
//#define ExitMode 0x61 //退出命令模式
//#define ReadFlag 0xf0 //读标志
//#define EnterSleep 0xff //进入睡眠模式
ram address read only RAM 地址(只读)
//#define AbmientTempAddr 0x03 //周围温度
//#define IR1Addr 0x04
//#define IR2Addr 0x05
//#define LineAbmientTempAddr 0x06 //环境温度
//
///*0x0000 0x4074 16500 0.01/单元
// -40 125*/
//#define LineObj1TempAddr 0x07 //目标温度,红外温度
///*0x27ad-0x7fff 0x3559 22610 0.02/单元
// -70.01-382.19 0.01 452.2*/
//#define LineObj2TempAddr 0x08
eepom address EEPROM 地址
//#define TObjMaxAddr 0x00 //测量范围上限设定
//#define TObjMinAddr 0x01 //测量范围下限设定
//#define PWMCtrlAddr 0x02 //PWM 设定
//#define TaRangeAddr 0x03 //环境温度设定
//#define KeAddr 0x04 //频率修正系数
//#define ConfigAddr 0x05 //配置寄存器
//#define SMbusAddr 0x0e //器件地址设定
//#define Reserverd1Addr 0x0f //保留
//#define Reserverd2Addr 0x19 //保留
//#define ID1Addr 0x1c //ID 地址 1
//#define ID2Addr 0x1d //ID 地址 2
//#define ID3Addr 0x1e //ID 地址 3
//#define ID4Addr 0x1f //ID 地址 4
/*
mlx90614各个端口定义
*/
sbit SDA=P1^6; //数据线
sbit SCK=P1^7; //时钟线
bdata uchar flag1; //可位寻址数据
sbit bit_out=flag1^7;
sbit bit_in=flag1^0;
/*
蜂鸣器引脚定义
*/
sbit BEEP = P2^3;
/*
定义1302时钟引脚以及1602液晶屏引脚 液晶屏数据引脚为P0
*/
sbit IO = P1^0;
sbit SCLK = P1^1;
sbit RST = P1^2;
sbit RS = P2^0;
sbit RW = P2^1;
sbit EN = P2^2;
/*
矩阵按键定义
*/
uchar Pre_KeyNO=16,KeyNO=16;
/*
定义时间和温度 数据缓冲数组 温度应为浮点型数据,但为了更好的输出将温度分为整数部分和小数部分,作为整形分别存储
*/
uchar *WEEK[]=
{
"SUN","MON","TUS","WEN","THU","FRI","SAT"
};
uchar LCD_DSY_BUFFER0[]={" "};
uchar LCD_DSY_BUFFER1[]={"DATE 00-00-00 "};
uchar LCD_DSY_BUFFER2[]={"TIME 00:00:00 "};
uchar LCD_DSY_BUFFER3[]={"Temperture:00.00"};
uchar DateTime[7];
uchar DateTem[2]={36,90};
uchar DateTem_real[2];
uchar tempH,tempL,err;
uchar DateBase[10]={36,80,37,90,35,70,36,30,35,80};
uchar DateBase_real[10]={36,80,37,90,35,70,36,30,35,80};
uchar number=0;
uchar number2=0;
/*
延时函数us级
*/
void delay(uint n)
{
uint j;
for(j=0;j<n;j++)
{
_nop_();
}
}
/*
延时函数
*/
void DelayMS(uint ms)
{
uchar i;
while(ms--)
{
for(i=0;i<120;i++);
}
}
/*
键扫描程序
*/
void Keys_Scan()
{
uchar Tmp;
P3 = 0x0f;
DelayMS(1);
Tmp = P3 ^ 0x0f;
switch(Tmp)
{
case 1: KeyNO = 0; break;
case 2: KeyNO = 1; break;
case 4: KeyNO = 2; break;
case 8: KeyNO = 3; break;
default: KeyNO = 16;
}
P3 = 0xf0;
DelayMS(1);
Tmp = P3 >> 4 ^ 0x0f;
switch(Tmp)
{
case 1: KeyNO += 0; break;
case 2: KeyNO += 4; break;
case 4: KeyNO += 8; break;
case 8: KeyNO += 12;
}
}
/*
蜂鸣器工作程序
*/
void Beep()
{
uchar i;
for(i=0;i<100;i++)
{
DelayMS(1);
BEEP = ~BEEP;
}
BEEP = 1;
}
/*
按键监听程序
*/
void key_listen()
{
P3=0xf0;
if(P3!=0xf0)
Keys_Scan();
if(Pre_KeyNO!=KeyNO)
{
Beep();
Pre_KeyNO=KeyNO;
}
DelayMS(100);
}
/*
90614的起始位子程序
*/
void start() //开始条件是 SCK=0 时,SDA 由 1 到 0
{
SDA=1;
delay(4);
SCK=1;
delay(4);
SDA=0;
delay(4);
SCK=0;
delay(4);
}
/*
90614的结束子程序
*/
void stop() //停止条件是 SCK=1 时,SDA 由 0 到 1
{
SCK=0;
delay(4);
SDA=0;
delay(4);
SCK=1;
delay(4);
SDA=1;
}
/*
90614位传送子程序
*/
void send_bit()
{
if(bit_out==1)
{
SDA=1; //发 1
}
else
{
SDA=0; //发 0
}
_nop_();
SCK=1; //上升沿
delay(4);
SCK=0;
delay(4);
}
/*
90614位接收子程序
*/
void read_bit()
{
SDA=1; //数据端先置 1
bit_in=1;
SCK=1; //上升沿
delay(4);
bit_in=SDA; //读数据
_nop_();
SCK=0;
delay(4);
}
/*
90614字节传送子程序
*/
void SendByte(uchar number)
{
uchar i,n,dat;
n=Nack_number; //可以重发次数
Send_again:
dat=number;
for(i=0;i<8;i++) //8 位依次发送
{
if(dat&0x80) //取最高位
{
bit_out=1; //发 1
}
else
{
bit_out=0; //发 0
}
send_bit(); //发送一个位
dat=dat<<1; //左移一位
}
read_bit(); //接收 1 位 应答信号
if(bit_in==1) //无应答时重发
{
stop();
if(n!=0)
{
n--; //可以重发 Nack_number=10 次
goto Repeat; //重发
}
else
{
goto exit; //退出
}
}
else
{
goto exit;
}
Repeat:
start(); //重新开始
goto Send_again; //重发
exit: ; //退出
}
/*
90614字节接收子程序
*/
uchar ReadByte()
{
uchar i,dat;
dat=0; //初值为 0
for(i=0;i<8;i++)
{
dat=dat<<1; //左移
read_bit(); //接收一位
if(bit_in==1)
{
dat=dat+1; //为 1 时对应位加 1
}
}
SDA=0; //发送应答信号 0
send_bit();
return dat; //带回接收数据
}
/*
90614读取温度子程序
*/
void readtemp()
{
SCK=0;
start(); //开始条件
SendByte(0x00); //发送从地址 00
SendByte(0x07); //发送命令
start(); //开始条件
SendByte(0x01); //读从地址 00
bit_out=0;
tempL=ReadByte(); //读数据低字节
bit_out=0;
tempH=ReadByte(); //读数据高字节
bit_out=1;
err=ReadByte(); //读错误信息码
stop(); //停止条件
if(DateTem_real[0]!=tempH && DateTem_real[1]!=tempL)
{
DateBase_real[number]=tempH;
DateBase_real[number+1]=tempL;
number++;
if(number==9)
{
number=0;
}
}
DateTem_real[0]=tempH;
DateTem_real[1]=tempL;
}
/*
向1302写数据
*/
void Write_A_Byte_TO_DS1302(uchar x)
{
uchar i;
for(i=0;i<8;i++)
{
IO=x&0x01;SCLK=1;SCLK=0;x>>=1;
}
}
/*
获取1302内部存储的时间数据
*/
uchar Get_A_Byte_FROM_DS1302()
{
uchar i,b=0x00;
for(i=0;i<8;i++)
{
b |= _crol_((uchar)IO,i);
SCLK=1;SCLK=0;
}
return b/16*10+b%16;
}
/*
根据地址逐一提取保存在1302内的时间变量
*/
uchar Read_Data(uchar addr)
{
uchar dat;
RST = 0;SCLK=0;RST=1;
Write_A_Byte_TO_DS1302(addr);
dat = Get_A_Byte_FROM_DS1302();
SCLK=1;RST=0;
return dat;
}
/*
以数组的形式将读取的时间保存
*/
void GetTime()
{
uchar i,addr = 0x81;
for(i=0;i<7;i++)
{
DateTime[i]=Read_Data(addr);addr+=2;
}
}
/*
读取1602的工作状态
*/
uchar Read_LCD_State()
{
uchar state;
RS=0;RW=1;EN=1;DelayMS(1);
state=P0;
EN = 0;DelayMS(1);
return state;
}
/*
1602检测忙等待,否则执行命令
*/
void LCD_Busy_Wait()
{
while((Read_LCD_State()&0x80)==0x80);
DelayMS(5);
}
/*
向1602写数据
*/
void Write_LCD_Data(uchar dat)
{
LCD_Busy_Wait();
RS=1;RW=0;EN=0;P0=dat;EN=1;DelayMS(1);EN=0;
}
/*
向1602写命令
*/
void Write_LCD_Command(uchar cmd)
{
LCD_Busy_Wait();
RS=0;RW=0;EN=0;P0=cmd;EN=1;DelayMS(1);EN=0;
}
/*
初始化1602
*/
void Init_LCD()
{
Write_LCD_Command(0x38);
DelayMS(1);
Write_LCD_Command(0x01);
DelayMS(1);
Write_LCD_Command(0x06);
DelayMS(1);
Write_LCD_Command(0x0c);
DelayMS(1);
}
/*
设置1602的显示坐标
*/
void Set_LCD_POS(uchar p)
{
Write_LCD_Command(p|0x80);
}
/*
1602显示字符串
*/
void Display_LCD_String(uchar p,uchar *s)
{
uchar i;
Set_LCD_POS(p);
for(i=0;i<16;i++)
{
Write_LCD_Data(s[i]);
DelayMS(1);
}
}
/*
将时间数据转化为逐一输出形式,逐一改变液晶屏相应位置的显示数据
*/
void Format_DateTime(uchar d,uchar *a)
{
a[0]=d/10+'0';
a[1]=d%10+'0';
}
void Format_DateTem(uchar d,uchar *a)
{
a[0]=d/10+'0';
a[1]=d%10+'0';
}
/*
显示时间
*/
void Show_Time()
{
GetTime();
Format_DateTime(DateTime[6],LCD_DSY_BUFFER1+5);
Format_DateTime(DateTime[4],LCD_DSY_BUFFER1+8);
Format_DateTime(DateTime[3],LCD_DSY_BUFFER1+11);
strcpy(LCD_DSY_BUFFER1+13,WEEK[DateTime[5]]);
Format_DateTime(DateTime[2],LCD_DSY_BUFFER2+5);
Format_DateTime(DateTime[1],LCD_DSY_BUFFER2+8);
Format_DateTime(DateTime[0],LCD_DSY_BUFFER2+11);
Display_LCD_String(0x00,LCD_DSY_BUFFER1);
Display_LCD_String(0x40,LCD_DSY_BUFFER2);
}
/*
显示温度
*/
void Show_Tem()
{
GetTime();
Format_DateTime(DateTime[2],LCD_DSY_BUFFER2+5);
Format_DateTime(DateTime[1],LCD_DSY_BUFFER2+8);
Format_DateTime(DateTime[0],LCD_DSY_BUFFER2+11);
Format_DateTem(DateTem[0],LCD_DSY_BUFFER3+11);
Format_DateTem(DateTem[1],LCD_DSY_BUFFER3+14);
Display_LCD_String(0x00,LCD_DSY_BUFFER2);
Display_LCD_String(0x40,LCD_DSY_BUFFER3);
}
/*
液晶屏清屏函数
*/
void Clear_Lcd()
{
Display_LCD_String(0x00,LCD_DSY_BUFFER0);
Display_LCD_String(0x40,LCD_DSY_BUFFER0);
}
void main()
{
Init_LCD(); //初始化1602
Show_Time();
DelayMS(1000);
Clear_Lcd();
while(1)
{
readtemp(); //读取温度 保存到真实数组中,演示采用程序中定义好的温度
Show_Tem();
key_listen();
if(Pre_KeyNO==0)
{
Clear_Lcd();
while(Pre_KeyNO!=3)
{
Format_DateTem(DateBase[number2],LCD_DSY_BUFFER3+11);
Format_DateTem(DateBase[number2+1],LCD_DSY_BUFFER3+14);
Display_LCD_String(0x40,LCD_DSY_BUFFER3);
key_listen();
if(Pre_KeyNO==1)
{
number2=number2+2;
Pre_KeyNO=16;
KeyNO=16;
}
if(Pre_KeyNO==2)
{
number2=number2-2;
Pre_KeyNO=16;
KeyNO=16;
}
if(number2==-2 || number2==10)
{
number2=0;
}
}
}
}
}
四、仿真图
五、硬件原理图
这个程序实现的功能:首先进入主函数,进行LCD1602的初始化,运行Show_Time函数,液晶屏第一行显示实时日期,第二行显示实时时间;延时一秒以后清屏然后液晶屏第一行显示实时时间,第二行显示温度(由于没有硬件,此温度为程序设定好的数组中的温度);这个程序也添加了按键的控制功能,按键用来作为模式选择,按键1作为模式选择键(按下时进入查看温度记录模块),按键2、按键3分别为下一个、上一个(实现对保存的记录的查看,总共会保存5组记录),在按键4按下时,退出查看温度记录模式,返回测温以及显示模式。
(含原理图、电路仿真图)