基于STM32环境温湿度采集

基于STM32环境温湿度采集

一、用到的器件

STM32F103C8T6 单片机
0.96 OLED
DHT11 温湿度传感器
4.7K 电阻(没有特别要求,普通就行)

二、实现的功能是

通过 DHT11 温湿度传感器采集现场温湿度,并显示在 OLED 屏幕上

三、原理图如下

DHT11 电路原理图
在这里插入图片描述

四、代码实现
4.1 DHT11 初始化函数

说明:开机默认输出模式,保持高电平。 DHT11 开机后会有 1S 左右的不稳定时间,因此在初
始化时有必要加入延时,跳过这一阶段, 这里设置延时为 1.5S。

/**
 * 功能:初始化DHT11 GPIO
 * 参数:None
 * 返回值:采集到的字节数据
 */
void initDHT11(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(DATA_LINE_CLOCK, ENABLE);
    
    GPIO_InitStructure.GPIO_Pin = DATA_LINE_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(DATA_LINE_PORT,&GPIO_InitStructure);
	
    GPIO_SetBits(DATA_LINE_PORT,DATA_LINE_PIN);

    Delay_ms(1500);                                     //开机稳定1.5S
}
4.2设置数据线输入输出函数

说明:数据线在发起采集开始信号时要设置为输出模式,采集过程要设置为输入,
这点和 IIC比较像,函数实现上也和 IIC 章节对应函数类似。

/**
 * 功能:设置数据线为输入
 * 参数:None
 * 返回值:None
 */
static void setLineIn(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    
    GPIO_InitStructure.GPIO_Pin = DATA_LINE_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
    
    GPIO_Init(DATA_LINE_PORT,&GPIO_InitStructure);
}

/**
 * 功能:设置数据线为输出
 * 参数:None
 * 返回值:None
 */
static void setLineOut(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;

    GPIO_InitStructure.GPIO_Pin = DATA_LINE_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    
    GPIO_Init(DATA_LINE_PORT,&GPIO_InitStructure);
    GPIO_SetBits(DATA_LINE_PORT,DATA_LINE_PIN);
}
4.3 采集单字节函数

说明:读取一个字节函数是整个读取过程的基础。
因为DHT11 数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器,
所以根据DHT11规则来写

/**
 * 功能:读取单个字节,整个读取流程需要调用该函数4次
 * 参数:None
 * 返回值:采集到的字节数据
 */
static u8 readByte(void)
{
    u8 i,byte = 0;
	u8 j = 0;

    for(i=0;i<8;++i)
    { 
        while(!GPIO_ReadInputDataBit(DATA_LINE_PORT,DATA_LINE_PIN) && ++j<10)      //等待低电平结束
		{
			Delay_us(10);
		}
		j = 0;
		
        Delay_us(30);                                                             //延时 用于后续判断
        if(GPIO_ReadInputDataBit(DATA_LINE_PORT,DATA_LINE_PIN))                   //延时后为高电平则为二进制1
        { 
            byte <<= 1;
            byte |= 0x01;
        }else 
        { 
            byte <<= 1;
        }

        while(GPIO_ReadInputDataBit(DATA_LINE_PORT,DATA_LINE_PIN) && ++j<10)      //等待剩余高电平
		{
			Delay_us(10);
		}
    }
    return byte;    
}
4.4 读取温湿度

说明:如果采集失败则返回 0xFFFF, 0xFFFF 是个无效数据,因为任何情况下温湿度都不能等于此值(255) , 所以可以依据此值进行错误数据判断。

/**
 * 功能:读取DHT11采集到的数据
 * 参数:
 *        mode:返回数据类型 用户可以指定返回温湿度当中的某一个或者所有
 *              返回所有数据时,湿度在高八位,温度在低八位
 * 返回值:采集到的字节数据
 *        
 */
u16 readDHT11(void)
{
    u8 i = 0;
    
    u8 Hum_H,Hum_L,Temp_H,Temp_L,CheckByte;                                 //依次是湿度整数、小数部分、温度整数、小数部分、校验和字节
    
    setLineOut();                                                           //发起读取开始信号
    LINE_LOW();
    Delay_ms(25);
    LINE_HIGH();

    setLineIn();

    while(GPIO_ReadInputDataBit(DATA_LINE_PORT,DATA_LINE_PIN) && ++i<10)   //主机释放总线空闲阶段
    {
        Delay_us(10);
    }
    i = 0;
    while(!GPIO_ReadInputDataBit(DATA_LINE_PORT,DATA_LINE_PIN) && ++i<10)  //传感器响应阶段
    {
        Delay_us(10);
    }
    i = 0;
    while(GPIO_ReadInputDataBit(DATA_LINE_PORT,DATA_LINE_PIN) && ++i<10)   //传感器拉高通知主机准备阶段
    {
        Delay_us(10);
    }

    Hum_H = readByte();                                                    //读取有效湿度值
    Hum_L = readByte();                                                    //湿度小数部分为0
    Temp_H = readByte();                                                   //读取有效温度值 
    Temp_L = readByte();                                                   //温度小数部分为0
    CheckByte = readByte();                                                //校验和                         

    if(Hum_H+Hum_L+Temp_H+Temp_L==CheckByte)                               //计算校验和
    {
        return Hum_H<<8 | Temp_H;

    }else 
    {		
        return 0xFFFF;                                                     //返回无效值,标识采集有误                                                
    }
        
}
4.5 主函数

说明:在采集温湿度的时候需要注意,官方手册建议采集间隔为 5S 以上,这里我们设置的采
集间隔是 2S,小于 2S 时所采集的数据会间歇性的返回 0xFFFF。

#include "DELAY/Delay.h"
#include "ADC/ADC.h"
#include "LED/LED.h"
#include "IIC/IIC.h"
#include "OLED/OLED.h"
#include "DHT11/DHT11.h"

int main(void)
{
	u8 i = 0;
	u16 data;
	/*初始化各外设*/ 
    initSysTick();  
	initADC();
	initLED();
    initDHT11();
	initIIC();
    initOLED();
	formatScreen(0x00);
    for(i=0;i<=128;i+=2)  //风媒电子LOGO从左向右动态显示
    {
		showImage(0,0,i,8,FM_LOGO_ENUM);
    }
	formatScreen(0x00);
    showCNString(32,0,"风媒电子",FONT_16_CN);
    showString(0,2,"Hum :",FONT_16_EN);
    showString(0,4,"Temp:",FONT_16_EN);
    showString(0,6,"Lux :",FONT_16_EN);
    while (1)
    {	
		if(++i>4)		//每隔2S采集一次 建议采集间隔最小不要小于2S
		{
			i = 0;
			data = readDHT11();
		}
        showNumber(40,2,data>>8,DEC,3,FONT_16_EN);
        showNumber(40,4,data&0x00FF,DEC,3,FONT_16_EN);
        showNumber(40,6,getConvValueAve(5,1000),DEC,4,FONT_16_EN);

        toggleLED();
        Delay_ms(500);

    }
}
  • 6
    点赞
  • 135
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值