关于单片机DTH11温湿度丈量仿真,曩昔大多无法进行,都用的SHT11来替代,但实际上,DHT11的驱运程序与SHT11的驱运程序有着本质上的差异,DHT11只需要接线3根线就能够作业,VCC、GND、DATA,作业时由单片机经过DATA线向DHT11发送发动信号,DHT11接到信号后回来一个应对信号,单片机收到到应对信号便可开端接纳温湿度数据(8bit湿度整数+8bit湿度小数+8bit温度整数+8bit温度小数+8bit校验和),SHT11是四线驱动。价格DHT11优胜于SHT11,读取上SHT11会愈加便利一些,SHT11,是四线驱动,直接4根插针数据时钟电源+ – 。DHT11和SHT11是同一个系列的,只不过它的丈量精度不同。后者的丈量精度比较高一些!现在,两款的温度湿度传器在PROTEUS中都能够进行仿真的。下面是仿真图。源程序及仿真。
源程序:
#include “REGX51.H”
#define LCD_DB P0
unsigned char s1[5];
unsigned char s2[5];
sbit LCD_RS=P0^7;
sbit LCD_RW=P0^6;
sbit LCD_E=P0^5;
sbit io = P1^0;
sbit moshi=P3^2;
sbit INC=P3^3;
sbit DEC=P3^4;
sbit SPK=P3^5;
#define uchar unsigned char
#define uint unsigned int
typedef unsigned charU8;
typedef unsigned intU16;
uchar count1=70,count2=35;
uchar moshicount=0;
uchar TD=0;//守时次数
U8U8FLAG;
U8U8count,U8temp;
U8U8T_data_H,U8T_data_L,U8RH_data_H,
U8RH_data_L,U8checkdata;
U8U8T_data_H_temp,U8T_data_L_temp,
U8RH_data_H_temp,U8RH_data_L_temp,U8checkdata_temp;
U8U8comdata;
void LCD_init(void);//初始化函数
void LCD_write_command(uchar command); //写指令函数
void LCD_write_data(uchar dat);
//写数据函数
void LCD_disp_char(uchar x,
uchar y,uchar dat);
//在某个屏幕方位上显现一个字符,X(0-16),y(1-2)
void delay_n40us(uint n);//延时函数
//********************************
//*******液晶初始化函数***************
void LCD_init(void)
{
LCD_write_command(0x38);
//设置 8 位格局,2 行,5×7
LCD_write_command(0x0c);
//全体显现,关光标,不闪耀
LCD_write_command(0x06);
//设定输入方法,增量不移位
LCD_write_command(0x03);
//铲除屏幕显现
delay_n40us(100);
}
//********************************
//*******守时器初始化函数**********
void TImerinit()
{
TMOD=0x01;
TH0=-50000/256;
TL0=-50000%6;
EA=1;
ET0=1;
}
//********************************
//********写指令函数************
void LCD_write_command(uchar dat)
{
LCD_DB=dat;
LCD_RS=0;//指令
LCD_RW=0;//写入
LCD_E=1;//答应
delay_n40us(1);
LCD_E=0;
delay_n40us(1);
}
//*******************************
//********写数据函数*************
void LCD_write_data(uchar dat)
{
LCD_DB=dat;
LCD_RS=1;//数据
LCD_RW=0;//写入
LCD_E=1;//答应
delay_n40us(1);
LCD_E=0;
delay_n40us(1);
}
//********************************
//*******显现一个字符函数*********
void LCD_disp_char(uchar x,
uchar y,uchar dat)
{
uchar address;
if(y==1)
address=0x80+x;
else
address=0xc0+x;
LCD_write_command(address);
LCD_write_data(dat);
}
//********************************
//********延时函数***************
void delay_n40us(uint n)
{
uint i;
uchar j;
for(i=n;i》0;i–)
for(j=0;j《2;j++);
}
voidDelay_10us(void)
{
U8 i;
i–;
i–;
i–;
i–;
i–;
i–;
}
void Delay(U16 j)
{
U8 i;
for(;j》0;j–)
for(i=0;i《27;i++);
}
//*******一字节数据传送函数*********
voidCOM(void)
{
U8 i;
for(i=0;i《8;i++)
{
U8FLAG=2;
while((!io)&&U8FLAG++);
Delay_10us();
Delay_10us();
Delay_10us();
U8temp=0;
if(io)U8temp=1;
U8FLAG=2;
while((io)&&U8FLAG++);
//超时则跳出for循环
if(U8FLAG==1)break;
//判别数据位是0仍是1
// 假如高电平高过预订0高电平值则数据位为 1
U8comdata《《=1;
U8comdata|=U8temp;
}
}
void RH(void)
{
//主机拉低18ms
io=0;
Delay(180);
io=1;
//总线由上拉电阻拉高 主机延时20us
Delay_10us();
Delay_10us();
Delay_10us();
Delay_10us();
//主机设为输入判别从机呼应信号
io=1;
//判别从机是否有低电平呼应信号如不呼应则跳出,呼应则向下运转
if(!io)//T !
{
U8FLAG=2;
//判别从机是否宣布 80us 的低电平呼应信号是否完毕
while((!io)&&U8FLAG++);
U8FLAG=2;
//判别从机是否宣布 80us 的高电平,如宣布则进入数据接纳状况
while((io)&&U8FLAG++);
//数据接纳状况
COM();
U8RH_data_H_temp=U8comdata;
COM();
U8RH_data_L_temp=U8comdata;
COM();
U8T_data_H_temp=U8comdata;
COM();
U8T_data_L_temp=U8comdata;
COM();
U8checkdata_temp=U8comdata;
io=1;
//数据校验
U8temp=(U8T_data_H_temp+U8T_data_L_temp+U8RH_data_H_temp+U8RH_data_L_temp);
if(U8temp==U8checkdata_temp)
{
U8RH_data_H=U8RH_data_H_temp;U8RH_data_L=U8RH_data_L_temp;
U8T_data_H=U8T_data_H_temp;
U8T_data_L=U8T_data_L_temp;
U8checkdata=U8checkdata_temp;
}
//湿度整数部分
s1[0] = (char)(0X30+U8RH_data_H/10);
s1[1] = (char)(0X30+U8RH_data_H);
//湿度小数部分
s1[2] = (char)(0X30+U8RH_data_L/10);
//温度整数部分
s2[0] = (char)(0X30+U8T_data_H/10);
s2[1] = (char)(0X30+U8T_data_H);
//温度小数部分
s2[2] = (char)(0X30+U8T_data_L/10);
}
}
//*******************************
//液晶显现函数
void disp()
{
LCD_disp_char(0,1,‘s’);
LCD_disp_char(1,1,‘h’);
LCD_disp_char(2,1,‘i’);
LCD_disp_char(3,1,‘d’);
LCD_disp_char(4,1,‘u’);
LCD_disp_char(5,1,‘:’);
LCD_disp_char(6,1,s1[0]);
LCD_disp_char(7,1,s1[1]);
LCD_disp_char(8,1,‘。’);
LCD_disp_char(9,1,s1[2]);
LCD_disp_char(10,1,‘%’);
LCD_disp_char(11,1,‘R’);
LCD_disp_char(12,1,‘H’);
LCD_disp_char(0,2,‘w’);
LCD_disp_char(1,2,‘e’);
LCD_disp_char(2,2,‘n’);
LCD_disp_char(3,2,‘d’);
LCD_disp_char(4,2,‘u’);
LCD_disp_char(5,2,‘:’);
LCD_disp_char(6,2,s2[0]);
LCD_disp_char(7,2,s2[1]);
LCD_disp_char(8,2,‘。’);
LCD_disp_char(9,2,s2[2]);
LCD_disp_char(10,2,0xDF);
LCD_disp_char(11,2,‘C’);
}
//阈值设置函数
void shezhi()
{
//初值
s1[3] = (char)(0X30+count1/10);
s1[4] = (char)(0X30+count1);
//初值
s2[3] = (char)(0X30+count2/10);
s2[4] = (char)(0X30+count2);
moshi=1;
if(moshi==0)
{
Delay_10us();
while(moshi==0);
moshicount++;
}
switch(moshicount)
{
case 1:
{
INC=1;DEC=1;
if(INC==0)
{
Delay_10us();
while(INC==0);
count1++;
}
else if(DEC==0)
{
Delay_10us();
while(DEC==0);
count1–;
}
LCD_disp_char(14,1,s1[3]);
LCD_disp_char(15,1,s1[4]);
LCD_disp_char(14,2,s2[3]);
LCD_disp_char(15,2,s2[4]);
}break;
case 2:
{
//moshicount=0;
INC=1;DEC=1;
if(INC==0)
{
Delay_10us();
while(INC==0);
count2++;
}
else if(DEC==0)
{
Delay_10us();
while(DEC==0);
count2–;
}
LCD_disp_char(14,1,s1[3]);
LCD_disp_char(15,1,s1[4]);
LCD_disp_char(14,2,s2[3]);
LCD_disp_char(15,2,s2[4]);
}break;
case 3:
{
moshicount=0;
LCD_disp_char(14,1,‘ ’);
LCD_disp_char(15,1,‘ ’);
LCD_disp_char(14,2,‘ ’);
LCD_disp_char(15,2,‘ ’);
}break;
default :break;
}
}
//蜂鸣器报警程序
void laba()
{
if((U8RH_data_H》=count1)||(U8T_data_H》=count2))
SPK=1;
else
SPK=0;
}
//守时器0中止程序,每次守时50ms
TImer0() interrupt 1
{
TD++;
if(TD》=100)
{
TH0=-50000/256;
TL0=-50000%6;
TR0=0;
TD=0;
}
TH0=-50000/256;
TL0=-50000%6;
}
//*********主函数*****************
void main(void)
{
LCD_init();
TImerinit();
Delay(4);
while(1)
{
RH();
disp();
laba();
TR0=1;
while((TD》=1)&&(TD《=100))
{
shezhi();
}
}
}