51单片机+DHT11温湿度传感器+LCD1602显示(最好用的程序,没有之一)

      DHT11传感器原理很简单(我就不过多解释了),但使用中可能会遇到很多问题,问题的本质大部分出在了延时函数之中,传统的软件延时与实际相差太多,所以尽量不要用,我所用的晶振为11.0952MHZ,一个周期约等于1Ms。不多啰嗦,直接上程序

#include
#include         //包含所需要的机械延时
#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--);
}

你可能感兴趣的:(51单片机+DHT11温湿度传感器+LCD1602显示(最好用的程序,没有之一))