基于STM8的DHT11温湿度传感器的驱动代码设计

     基于STM8的DHT11温湿度传感器的驱动代码设计

  最近希望恢复性学习一下STM8的相关知识,于是我选择了从头开始写温湿度传感器DHT11驱动代码的方式。其中遇到一些问题,也有一些收获,希望会帮助到遇到类似问题的朋友,也希望不足之处得到大家的指导

      首先介绍一下DHT11的必要知识

     一 复位时序 以及 数据时序

  

   下面是数据时序

此外,根据数据手册得知,一次通信需要的时间是3毫秒左右,这很重要,在后面的BUG分析环节会说到

二 贴上关键代码以及分析

  

//复位DHT11
void DHT11_RST()
{  
   TIM4_CR1 = 0x00;                      //关闭定时器
   TIM4_CNTR = 0;                        //保证下次的第一个数据位的准确
   DATA_SET;                              //ODR设置为1
   DATA_OUT();                            //推挽输出模式,此时输出高电平
   DATA_CLR;                                //此时处于主机输出模式,总线拉低
   TIM2_Delayus(20000);                     //拉低20毫秒 
   DATA_SET;                                //释放总线 
   TIM2_Delayus(40);                     //释放总线以后等待40微秒DHT会发出响应信号
}
扫描二维码关注公众号,回复: 3869976 查看本文章
//检测DHT11是否响应
uchar DHT11_CHECK()
{
  
  if(!DATA_GET)                       //如果顺利拉低,就说明有了响应
  {
  while((!DATA_GET)&&(outline<100))                    //先是低电平
  {
  TIM2_Delayus(1);
  }
  if(outline>90)                                       //起始信号超时退出
  return 0;
  outline = 0;
  while((DATA_GET)&&(outline<100))                    //接着是高电平
  {
  TIM2_Delayus(1);
  }
  if(outline<90)
  TIM4_CR1 = 0x81;                                    //立刻打开定时器开始计时第一个数据位
  else
  return 0;
  DATA_IN();                                          //引脚设置为外部中断模式
  outline = 0;      
  
  return 1;                                          //一切成功返回1
  
  }
  else
  return 0;
  
}
#pragma vector = 0x05                               //PA的中断向量位
__interrupt void GPIOA_IRQHandler()
{
   datatime = TIM4_CNTR;         //获取两次下降沿之间的数据宽度
   TIM4_CNTR = 0;                //清零,再次获取下一位
   datareg <<= 1;                //高位先出,左移操作
   if((datatime>75)&&(datatime<85))           //数据0  我就默认高位开始获取了
   datareg &= 0xfe;                        
   if((datatime>120)&&(datatime<130))         //数据1
   datareg |= 0x01;                       
   if(datanum == 7)
   dataall[0] = datareg;                        //获取第一个字节也就是湿度整数位
   if(datanum == 23)                            //获取第三个字节也就是温度整数位
   dataall[1] = datareg;
   if(datanum == 39)                            //获取第五个字节也就是校验(温度+湿度)位
   dataall[2] = datareg;
   datanum++;                                 //每次读取一位进1
   if(datanum >= 40)                          //数据接收完了结束
   datanum = 0;
   
}

三 总结以及BUG分析

   总的来说 这是一款使用起来非常简单的传感器,但是作为菜鸟的我依旧是遇到了好多的问题

BUG 1  Q:  复位完毕以后,DHT11拉低总线然后再度拉高之后就不再拉低,不出数据

             A:  因为在之前的程序中,我喜欢在DHT拉低以后用串口发送一个"0 FINISH"来标记DHT的引脚响应情况,而且这样也显得很叼。可是之前说过了,一次DHT的数据通信大概就3毫秒,可是你知道串口发送字符串是一件多么努力而且费时间的事情吗,你把人家DHT最好的年华都错过了啊,当你再次读取高电平的时候,对不起,这已经是数据通信结束的事情了。所以,单总线时序中不要加入一些影响读取时序的代码。

BUG 2  Q:用下降沿获取数据位数的时候,发现触发非常多,而且无论如何修改触发方式都无法改变这一现状

             A:这里要说到一个之前不知道的小知识,EXTI_CR寄存器只有在总中断关闭的是时候才可以修改,所以之前一直无法修改,默认的进行了下降沿以及低电平触发的方式。当然失败了。至于其他寄存器是不是也这样就不得而知了。在之后的学习中会慢慢记住的。

好了本菜鸟的心路历程记录完了。也可以关注我自己的微信公众号Haer_MCU,那里面和这里也差不多,都是一些自己做完感觉很喜欢的小尝试。本人的完整代码稍后也会上传。

资源已经上传

http://download.csdn.net/detail/haer_mcu/9688014

这里是地址

猜你喜欢

转载自blog.csdn.net/qq_34744658/article/details/83587326