STM32读取温湿度传感器DHT11和DHT21(AM2301)系列问题

时间:2024-01-23 09:55:01

1、DHT11和DHT21传感器

    这两种传感器都是奥松公司的产品,具体的传感器说明书在其官网上有(www.aosong.com)。

    DHT11 数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器。它应用专用的数 字模块采集技术和温湿度传感技术,确保产品具有枀高的可靠性与卓越的长期稳定性。传感器包括一 个电容式感湿元件和一个 NTC 测温元件,并与一个高性能 8 位单片机相连接。

    DHT21(AM2301)湿敏电容数字温湿度模块是一款含有己校准数字信号输出的温湿度复合传感器。它应用专用的数字模块采集技术和温湿度传感技术,确保产品具有极高的可靠性与卓越的长期稳定性。传感器包括一个电容式感湿元件和一个高精度测温元件,并与一个高性能 8 位单片机相连接。因此该产品具有品质卓越、超快响应、抗干扰能力强、性价比极高等优点。每个传感器都在极为精确的湿度校验室中进行校准。校准系数以程序的形式储存在单片机中,传感器内部在检测信号的处理过程中要调用这些校准系数。标准单总线接口,使系统集成变得简易快捷。超小的体积、极低的功耗,信号传输距离可达 20 米以上,使其成为各类应用甚至最为苛刻的应用场合的最佳选择。产品为 3 引线(单总线接口)连接方便。特殊封装形式可根据用户需求而提供。

2、温湿度读取方式

    两种传感器均采用简化的单总线通信。单总线即只有一根数据线,系统中的数据交换、控制均由单总线完 成。设备(主机或从机)通过一个漏极开路或三态端口连至该数据线,以允许设备在不发送数据时能够释 放总线,而让其它设备使用总线;单总线通常要求外接一个约 4.7kΩ的上拉电阻,这样,当总线闲置时, 其状态为高电平。由于它们是主从结构,只有主机呼叫从机时,从机才能应答,因此主机访问器件都必须 严格遵循单总线序列,如果出现序列混乱,器件将不响应主机。

3、两种传感器的主要区别

   包括两点,分别是单总线起始信号的数据总线拉低时间温湿度高低位数据含义

   DHT11的单总线格式定义:

   DHT21的单总线格式定义:

4、微秒级延时函数

    根据上面的介绍我们现在需要一个微秒级的精确延时函数,否则就不能成功的读取传感器数据。STM32单片机的滴答定时器可以满足。

  (1)保证滴答定时器的时钟频率为72MHz

    STM32F103内部8M的内部震荡,经过倍频后最高可以达到72M。在固件库3.0以上,只需要通过两步就可以完成对时钟频率的选择。

   ①修改system_stm32f10x.c开头的时钟宏定义,如下:

/* #define SYSCLK_FREQ_HSE    HSE_VALUE */
/* #define SYSCLK_FREQ_24MHz  24000000 */ 
/* #define SYSCLK_FREQ_36MHz  36000000 */
/* #define SYSCLK_FREQ_48MHz  48000000 */
/* #define SYSCLK_FREQ_56MHz  56000000 */
#define SYSCLK_FREQ_72MHz  72000000

   ②在主函数main中调用SystemInit()函数。

   ③如果没有成功,需要检查单片机的晶振是否起振。简单的检测方法:

       有示波器的可以查看晶振的输出波形,正常应为正弦波。没有示波器的可以利用万用表分别测量晶振的管脚电平,正常应在1/2*VCC左右。

  (2)程序编写

    ①在main()函数中添加:SysTick_Config(72000000 / 1000000);//除以1000000微秒级(100000十微秒级,1000毫秒级),指进入中断的间隔时间

int main(void)
{      
    SystemInit();
    SysTick_Config(72000000 / 1000000);
}

    ②Delay.c

#include "stm32f10x.h"
#include "Delay.h"

int TimingDelay=0;

void delay_us(u32 n)
{
    TimingDelay=n;
  while(TimingDelay!=0); 
}
void delay_ms(u32 n)
{
    while(n--)
    {
     delay_us(1000);        
    }
}

      Delay.h

#ifndef _DELAY__H_
#define _DELAY__H_

void delay_us(u32 n);
void delay_ms(u32 n);

#endif

    ③stm32f10x_it.c

extern int TimingDelay;
void SysTick_Handler(void)
{
    if(TimingDelay!=0)
    {
        TimingDelay--;    
    }
}

5、编写读传感器数据程序

    ①DHxx.c

#include "stm32f10x.h"
#include "DHxx.h"
#include "Delay.h"
#include "sys.h"
u8 tdata[4]={0x00,0x00,0x00,0x00};
u8 sbuf,check;

//***************************************************************************/
//      //uchar       i;
//      uchar    U8FLAG,k;
//      uchar    U8count,U8temp;
//      uchar    U8T_data_H,U8T_data_L,U8RH_data_H,U8RH_data_L,U8checkdata;
//      uchar    U8T_data_H_temp,U8T_data_L_temp,U8RH_data_H_temp,U8RH_data_L_temp,U8checkdata_temp;
//      uchar    U8comdata;
//      uint   ReceiveHighByte;
//      uint   ReceiveLowByte;
//***************Global defination for DHT11 end****// 
void DHT_PortIN(void)
{
    GPIO_InitTypeDef  GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 ;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;     
    GPIO_Init(GPIOB,&GPIO_InitStructure);
}
void DHT_PortOUT(void)
{
    GPIO_InitTypeDef  GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 ;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;     
    GPIO_Init(GPIOB,&GPIO_InitStructure);
}
unsigned char StartDHT11(void)
{
   DHT_PortOUT();
   DHT_Out = 0;
   delay_ms(20);  //DHT11>18ms
  
   DHT_Out = 1;    
     DHT_PortIN();
   delay_us(20);  

   if(!DHT_In)
   {
           while(!DHT_In);
           while(DHT_In);
           return 1;
   }
    return 0;
}

unsigned char StartDHT21(void)
{
   DHT_PortOUT();
   DHT_Out = 0;
   delay_ms(1);  //DHT21>800us
  
   DHT_Out = 1;
     DHT_PortIN();
   delay_us(20);  

   if(!DHT_In)
   {
            while(!DHT_In);
           while(DHT_In);
           return 1;
   }
    return 0;
}
void com(void)
{
    u8 i,tt;
    tt = 0;
    for(i = 0;i<8;i++)
    {
        sbuf <<= 1;
        delay_us(20);
        while(!DHT_In);
        delay_us(25);
        tt = 100;
        while(tt++);
        if(DHT_In)
        {
             sbuf |= 0x01;
             delay_us(30);
        }
        else
        {
            sbuf &= 0xfe;
        }
    }
}
u8 ReadDHT11(void)
{
    u8 sum;
    if(StartDHT11())    
    {
        com();
        tdata[0]=sbuf;
        com();
        tdata[1]=sbuf;
        com();
        tdata[2]=sbuf;        
        com();
        tdata[3]=sbuf;
        com();
        check = sbuf;
        sum = (tdata[0]+tdata[1]+tdata[2]+tdata[3]);
    }
    if(check == sum)
    return(1);
    else    
    return 0;
}

u8 ReadDHT21(void)
{
    u8 sum;
    if(StartDHT21())    
    {
        com();
        tdata[0]=sbuf;
        com();
        tdata[1]=sbuf;
        com();
        tdata[2]=sbuf;        
        com();
        tdata[3]=sbuf;
        com();
        check = sbuf;
        sum = (tdata[0]+tdata[1]+tdata[2]+tdata[3]);
    }
    if(check == sum)
    return(1);
    else    
    return 0;
}

    DHxx.h                     //(注意:我这边定义的DATA管脚为PB1)

#ifndef _DHXX__H_
#define _DHXX__H
#include "sys.h"
#define DHT_Out PBout(1)
#define DHT_In  PBin(1)

#define uchar unsigned char 
#define uint  unsigned int

unsigned char StartDHT11(void);
unsigned char StartDHT21(void);
void DHT_PortIN(void);
void DHT_PortOUT(void);
u8 ReadDHT11(void);
u8 ReadDHT21(void);
void com(void);
//**********************************

//**********************************

extern u8 tdata[4];
extern u8 sbuf;
extern u8 check;

#endif

    ②sys.h

#include "stm32f10x.h"

///////////////////////////////////////////////////////////////
#define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x2000000+((addr &0xFFFFF)<<5)+(bitnum<<2)) 
#define MEM_ADDR(addr)  *((volatile unsigned long  *)(addr)) 
#define BIT_ADDR(addr, bitnum)   MEM_ADDR(BITBAND(addr, bitnum)) 
//IO地址映射
#define GPIOA_ODR_Addr    (GPIOA_BASE+12) //0x4001080C 
#define GPIOB_ODR_Addr    (GPIOB_BASE+12) //0x40010C0C 
#define GPIOC_ODR_Addr    (GPIOC_BASE+12) //0x4001100C 
#define GPIOD_ODR_Addr    (GPIOD_BASE+12) //0x4001140C 
#define GPIOE_ODR_Addr    (GPIOE_BASE+12) //0x4001180C 
#define GPIOF_ODR_Addr    (GPIOF_BASE+12) //0x40011A0C    
#define GPIOG_ODR_Addr    (GPIOG_BASE+12) //0x40011E0C    

#define GPIOA_IDR_Addr    (GPIOA_BASE+8) //0x40010808 
#define GPIOB_IDR_Addr    (GPIOB_BASE+8) //0x40010C08 
#define GPIOC_IDR_Addr    (GPIOC_BASE+8) //0x40011008 
#define GPIOD_IDR_Addr    (GPIOD_BASE+8) //0x40011408 
#define GPIOE_IDR_Addr    (GPIOE_BASE+8) //0x40011808 
#define GPIOF_IDR_Addr    (GPIOF_BASE+8) //0x40011A08 
#define GPIOG_IDR_Addr    (GPIOG_BASE+8) //0x40011E08 
 
//IO口操作,只针对单一的IO口//n的值小于16
#define PAout(n)   BIT_ADDR(GPIOA_ODR_Addr,n)  //输出
#define PAin(n)    BIT_ADDR(GPIOA_IDR_Addr,n)  //输入

#define PBout(n)   BIT_ADDR(GPIOB_ODR_Addr,n)  
#define PBin(n) BIT_ADDR(GPIOB_IDR_Addr,n) #define PCout(n) BIT_ADDR(GPIOC_ODR_Addr,n) #define PCin(n) BIT_ADDR(GPIOC_IDR_Addr,n) #define PDout(n) BIT_ADDR(GPIOD_ODR_Addr,n) #define PDin(n) BIT_ADDR(GPIOD_IDR_Addr,n) #define PEout(n) BIT_ADDR(GPIOE_ODR_Addr,n) #define PEin(n) BIT_ADDR(GPIOE_IDR_Addr,n) #define PFout(n) BIT_ADDR(GPIOF_ODR_Addr,n) #define PFin(n) BIT_ADDR(GPIOF_IDR_Addr,n) #define PGout(n) BIT_ADDR(GPIOG_ODR_Addr,n) #define PGin(n) BIT_ADDR(GPIOG_IDR_Addr,n) /////////////////////////////////////////////////////////////////

    ③main.c

/* Includes ------------------------------------------------------------------*/
#include "stm32f10x.h"
#include <stdio.h>
#include "system.h"
#include "sys.h"
#include "DHxx.h"
#include "Delay.h"
int main(void)
{      
    SystemInit();
    SysTick_Config(72000000 / 1000000);
    while(1)
    {
        ReadDHT11();// ReadDHT21();//读取的信息在DHxx.c的tdata[]数组中
        delay_ms(1000);
        delay_ms(1000);  //注意读取间隔应该在2秒以上 
        }
}    

6、参考

【1】单片机stm32时钟频率 http://www.elecfans.com/dianzichangshi/20171113578546.html

【2】https://wenku.baidu.com/view/6306fbf484254b35eefd3428.html

【3】http://www.aosong.com/products-28.html

  

人不曾老去,直至悔恨取代了梦想!---约翰.巴里摩

 转载需说明出处,笔者总结之前的知识,与大家分享,有问题的可以留给我哦~