STM32驱动dht11温湿度传感器

DHT11 是一款湿温度一体化的数字传感器。该传感器包括一个电阻式测湿元件和一个 NTC
测温元件,并与一个高性能 8 位单片机相连接。通过单片机等微处理器简单的电路连接就能够
实时的采集本地湿度和温度。 DHT11 与单片机之间能采用简单的单总线进行通信,仅仅需要一
个 I/O 口。传感器内部湿度和温度数据 40Bit 的数据一次性传给单片机,数据采用校验和方式
进行校验,有效的保证数据传输的准确性。DHT11 功耗很低,5V 电源电压下,工作平均最大
电流 0.5mA。
DHT11 的技术参数如下:
  工作电压范围:3.3V -5.5V
  工作电流  :平均 0.5mA
  输出:单总线数字信号
  测量范围:湿度 20~90%RH,温度 0~50℃
  精度  :湿度±5%,温度±2℃

  分辨率  :湿度 1%,温度 1℃


DHT11有效总线包含三条,VCC GND DAT,看起来与DS18B20类似,但是简单很多,不需要设置命令,只需要读取数据包就可以了,

每次读取数据一共读取40个BIT也就是五个字节,高位在前MSB

五个字节分别是:  8bit湿度整数数据+8bit湿度小数数据+8bi温度整数数据+8bit温度小数数据   +8bit校验和 


读写时序如下

  首先主机发送开始信号,即:(最开始状态依然是高电平)拉低数据线,保持 t1 (至少 18ms)时间,然后拉高数据线 t2(20~40us)时间,(此时需要转换输入输出模式)然后读取 DHT11 的响应,正常的话, DHT11 会拉低数据线,保持 t3 (40~50us)时间,作为响应信号,然后 DHT11 拉高数据线,保持 t4(40~50us)时间后,开始输出数据

  也就是说,每次需要复位,检查响应,才能开始读数据,数据的格式如下

    由此我们可以看到,每个数据都是有一个12-14us的起始位开始,是0还是1需要我们监测之后的高电平时间长度,基本上我们可以认为高电平持续时间大于35us的基本就是1了

(注意不能等待这个电平超过40us)因为一次0的时间就是40us,等待太长会可能丢掉下一个数据的起始位(这里我们可以用等待点评延时计数的模式来判定时间,当电平为0,等待他为1,每等待一次计数1us,最后看高电平持续时间)


驱动代码如下所示


    
    
  1. #ifndef __DHT11_H
  2. #define __DHT11_H
  3. #include "ioremap.h"
  4. #include "delay.h"
  5. #include "uart.h"
  6. //IO方向设置
  7. #define DHT11_IO_IN() {GPIOG->CRH&=0XFFFF0FFF;GPIOG->CRH|=8<<12;}
  8. #define DHT11_IO_OUT() {GPIOG->CRH&=0XFFFF0FFF;GPIOG->CRH|=3<<12;}
  9. ////IO操作函数
  10. #define DHT11_DQ_OUT PGout(11) //数据端口 PG11
  11. #define DHT11_DQ_IN PGin(11) //数据端口 PG11
  12. u8 Dht11Init(void); //初始化DHT11
  13. u8 Dht11ReadData(u8 *temp,u8 *humi); //读取温湿度
  14. u8 Dht11ReadByte(void); //读出一个字节
  15. u8 Dht11ReadBit(void); //读出一个位
  16. u8 Dht11Check(void); //检测是否存在DHT11
  17. void Dht11Rst(void); //复位DHT11
  18. void Dht11Show(void);
  19. #endif

    
    
  1. #include "dht11.h"
  2. //复位DHT11
  3. void Dht11Rst(void)
  4. {
  5. DHT11_IO_OUT(); //SET OUTPUT
  6. DHT11_DQ_OUT= 0; //拉低DQ
  7. DelayMs( 20); //拉低至少18ms
  8. DHT11_DQ_OUT= 1; //DQ=1
  9. DelayUs( 30); //主机拉高20~40us
  10. }
  11. //等待DHT11的回应
  12. //返回1:未检测到DHT11的存在
  13. //返回0:存在
  14. u8 Dht11Check(void)
  15. {
  16. u8 retry= 0;
  17. DHT11_IO_IN(); //SET INPUT
  18. while (DHT11_DQ_IN&&retry< 100) //DHT11会拉低40~80us
  19. {
  20. retry++;
  21. DelayUs( 1);
  22. };
  23. if(retry>= 100) return 1;
  24. else retry= 0;
  25. while (!DHT11_DQ_IN&&retry< 100) //DHT11拉低后会再次拉高40~80us
  26. {
  27. retry++;
  28. DelayUs( 1);
  29. };
  30. if(retry>= 100) return 1;
  31. return 0;
  32. }
  33. //从DHT11读取一个位
  34. //返回值:1/0
  35. u8 Dht11ReadBit(void)
  36. {
  37. u8 retry= 0;
  38. while(DHT11_DQ_IN&&retry< 100) //等待变为低电平
  39. {
  40. retry++;
  41. DelayUs( 1);
  42. }
  43. retry= 0;
  44. while(!DHT11_DQ_IN&&retry< 100) //等待变高电平
  45. {
  46. retry++;
  47. DelayUs( 1);
  48. }
  49. DelayUs( 40); //等待40us
  50. if(DHT11_DQ_IN) return 1;
  51. else return 0;
  52. }
  53. //从DHT11读取一个字节
  54. //返回值:读到的数据
  55. u8 Dht11ReadByte(void)
  56. {
  57. u8 i,dat;
  58. dat= 0;
  59. for (i= 0;i< 8;i++)
  60. {
  61. dat<<= 1;
  62. dat|=Dht11ReadBit();
  63. }
  64. return dat;
  65. }
  66. //从DHT11读取一次数据
  67. //temp:温度值(范围:0~50°)
  68. //humi:湿度值(范围:20%~90%)
  69. //返回值:0,正常;1,读取失败
  70. u8 Dht11ReadData(u8 *temp,u8 *humi)
  71. {
  72. u8 buf[ 5];
  73. u8 i;
  74. Dht11Rst();
  75. if(Dht11Check()== 0)
  76. {
  77. for(i= 0;i< 5;i++) //读取40位数据
  78. {
  79. buf[i]=Dht11ReadByte();
  80. }
  81. if((buf[ 0]+buf[ 1]+buf[ 2]+buf[ 3])==buf[ 4])
  82. {
  83. *humi=buf[ 0];
  84. *temp=buf[ 2];
  85. }
  86. } else return 1;
  87. return 0;
  88. }
  89. //初始化DHT11的IO口 DQ 同时检测DHT11的存在
  90. //返回1:不存在
  91. //返回0:存在
  92. u8 Dht11Init(void)
  93. {
  94. RCC->APB2ENR|= 1<< 8; //使能PORTG口时钟
  95. GPIOG->CRH&= 0XFFFF0FFF; //PORTG.11 推挽输出
  96. GPIOG->CRH|= 0X00003000;
  97. GPIOG->ODR|= 1<< 11; //输出1
  98. Dht11Rst();
  99. return Dht11Check();
  100. }
  101. void Dht11Show(void)
  102. {
  103. u8 temp,humi;
  104. if(Dht11ReadData(&temp,&humi))
  105. {
  106. printf( "DHT11 read failed\r\n");
  107. }
  108. else
  109. {
  110. printf( "温度 %d 湿度 %d \r\n",temp,humi);
  111. }
  112. }




转载于:https://www.cnblogs.com/dengxiaojun/p/4279448.html

猜你喜欢

转载自blog.csdn.net/bxyBLUR/article/details/106796005