? ? ? DHT11传感器原理很简单(我就不过多解释了),但使用中可能会遇到很多问题,问题的本质大部分出在了延时函数之中,传统的软件延时与实际相差太多,所以尽量不要用,我所用的晶振为11.0952MHZ,一个周期约等于1Ms。不多啰嗦,直接上程序
#include<reg52.h>
#include<intrins.h> //包含所需要的机械延时
#define uint unsigned int
#define uchar unsigned char
/*******************************
*********dht11相关配置**********
***只需要直接调用数据接收函数****
*******************************/
sbit temp_date=P1^0; //可根据自己的接口自行修改
void dht11_start(); //让dht11开始发送数据
uchar rec_byte(); //返回一个字节 有返回值不要用void
void dht11_receive(); //dht11接收数据的函数,start与rec_byte()均包含在其中
/*****LCD1602相关函数*********
*RW接地即可,也可以配置为低电平*
**我用的P0端口,需要加上拉电阻**
*只需简单延时即可,无需复杂操作*
*****************************/
void write_com(uchar com); //写命令
void write_data(uchar date);//写数据
void lcd1602init();
void delayms(uint z);
sbit lcdrs=P3^5; //根据实际情况自行配置
sbit lcden=P3^4; //根据实际情况自行配置
/***********主函数**********/
void main()
{
delayms(1000);
delayms(1000); //上电后等待元器件启动
lcd1602init(); //1602初始化
while(1)
{
dht11_receive(); //接收温湿度数据其中已经让其在1602中显示,当然可以自行调节
delayms(100); //没100Ms检测一次
}
}
/*开始*/
void dht11_start()
{
temp_date=1;
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
temp_date=0;
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_();
temp_date=1;
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_();
}
/***********dht11相关子函数*************/
uchar rec_byte() //接收一个字节
{
uchar i,dat=0;
for(i=0;i<8;i++) //从高到低依次接收8位数据
{
while(!temp_date); //等待50us低电平过去
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
_nop_(); _nop_(); _nop_(); _nop_();
//延时40us,如果还为高则数据为1,否则为0
dat<<=1; //移位使正确接收8位数据,数据为0时直接移位
if(temp_date==1) //数据为1时,使dat加1来接收数据1
dat+=1;
while(temp_date); //等待数据线拉低
}
return dat;
}
void dht11_receive() //接收数据的函数并显示
{
uchar R_H,R_L,T_H,T_L,revise; //分别为湿度高低位,温度高低位,效验位
uchar RL,RH,TH,TL,sd1,sd2,wd1,wd2;//均使用的局部变量只能在这个复合函数中使用
//sd 湿度拼音缩写 wd 温度拼音缩写
dht11_start();
if(temp_date==0) //判断是否响应
{
while(temp_date==0);
while(temp_date);
/**接收数据**/
R_H=rec_byte();
R_L=rec_byte();
T_H=rec_byte();
T_L=rec_byte();
revise=rec_byte(); //准备检验
_nop_();_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();_nop_();
if(R_H+R_L+T_H+T_L==revise) //检验是否正确
{
RH=R_H;
RL=R_L;
TH=T_H;
TL=T_L;
}
/***************计算****************/
wd1=TH/10;
wd2=TH%10;
sd1=RH%100/10;
sd2=RH%10;
temp_date=1;
}
write_com(0x80); //第一行从开头写
write_data('w');
write_data('e');
write_data('n');
write_data('d');
write_data('u');
write_data('0'+wd1);
write_data('0'+wd2);
write_data(' ');
write_data(' ');
write_data('s');
write_data('h');
write_data('i');
write_data('d');
write_data('u');
write_data('0'+sd1);
write_data('0'+sd2);
}
/*********LCD1602相关子函数**********/
void write_com(uchar com)
{
lcdrs=0;
P0=com;
delayms(5);
lcden=1;
delayms(5);
lcden=0;
}
void write_data(uchar date)
{
lcdrs=1;
P0=date;
delayms(5);
lcden=1;
delayms(5);
lcden=0;
}
void lcd1602init()
{
lcden=0;
write_com(0x38);
write_com(0x0c);
write_com(0x06);
write_com(0x01);
}
/*************简单延时函数************/
void delayms(uint z)
{
uint x,y;
for(x=z;x>0;x--)
for(y=110;y>0;y--);
}
|