STM32F1基于标准库ST7735 1.8‘‘LCD显示DHT11数据

STM32基于标准库ST7735 1.8‘’LCD显示DHT11数据


🌻ST7735 128x160 1.8’'LCD屏幕

📓DHT11驱动代码

  • 🌿bsp_DHT11.h
/***********************************************************************************************************************************
 **【文件名称】  bsp_DHT11.c
 **【文件功能】
 **【适用平台】  STM32F103 + 标准库v3.5 + keil5

************************************************************************************************************************************/
#include "bsp_DHT11.h"

xDHT11_TypeDef       xDHT11;             // 声明全局结构体, 用于记录信息

static GPIO_TypeDef *DHT11_GPIOx = 0;    // 引脚端口
static uint32_t      DHT11_PINx  = 0;    // 引脚编号

#define  DHT11_BUS_HIGH     (DHT11_GPIOx->BSRR = (uint32_t)DHT11_PINx)           // DAT引脚 置高电平
#define  DHT11_BUS_LOW      (DHT11_GPIOx->BSRR = ((uint32_t)DHT11_PINx) << 16)   // DAT引脚 置低电平
#define  DHT11_BUS_READ     ((DHT11_GPIOx->IDR & DHT11_PINx) ? 1: 0)             // 读取引脚的电平



#if DELAY_MODE_TIM2
#include "stm32f10x_tim.h"
#include "misc.h"
// 定时器初始化配置
// 不使用中断
void Tim2_Init(uint16_t psc, uint16_t arr, FunctionalState NewState)
{
    TIM_TimeBaseInitTypeDef tim;                         // 结构体

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // 开启TIM时钟
    TIM_DeInit(TIM2);
    tim.TIM_ClockDivision = TIM_CKD_DIV1;                // 采样分频
    tim.TIM_CounterMode   = TIM_CounterMode_Up;          // 向上计数
    tim.TIM_Period        = arr;                         // 自动重装载寄存器的值
    tim.TIM_Prescaler     = psc - 1;                     // 时钟预分频
    TIM_TimeBaseInit(TIM2, &tim);                        // 初始化结构体
    TIM2->SR = (uint16_t)~((uint16_t)0x01);              // 清除更新标志 
    TIM2->CNT = 0;                                       // 清0计数器
    TIM_Cmd(TIM2, NewState);                             // 是否开始工作
}



static void delay_us(uint32_t us)
{
    Tim2_Init(72, us, ENABLE);
    while ((TIM2->SR & 1) == 0);
    TIM_Cmd(TIM2, DISABLE);
}



static void delay_ms(uint32_t ms)
{
    for (uint16_t i = 0; i < ms; i++)
    {
        Tim2_Init(72, 990, ENABLE);
        while ((TIM2->SR & 1) == 0);
        TIM_Cmd(TIM2, DISABLE);
    }
}
#else

// 本地US粗略延时函数,减少移植时对外部文件依赖;
static void delay_us(uint32_t us)
{
    uint16_t i = 0;
    while (us--)
    {
        i = 7;
        while (i--);
    }
}

// 本地MS粗略延时函数,减少移植时对外部文件依赖;
static void delay_ms(uint32_t ms)
{
    uint32_t i = 0;
    while (ms--)
    {
        i = 12000;
        while (i--);
    }
}
#endif // 结束延时



static void DHT11_Mode_IPU(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;

    GPIO_InitStructure.GPIO_Pin = DHT11_PINx ;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU ;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(DHT11_GPIOx, &GPIO_InitStructure);
}

static void DHT11_Mode_Out_PP(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = DHT11_PINx ;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(DHT11_GPIOx, &GPIO_InitStructure);
}

static void DHT11_Init(GPIO_TypeDef *GPIOx, uint32_t PINx)
{
    DHT11_GPIOx = GPIOx;
    DHT11_PINx = PINx;

    // 时钟使能:引脚端口;用判断端口的方式使能时钟线, 减少移植时的工作
    if (GPIOx == GPIOA)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    if (GPIOx == GPIOB)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    if (GPIOx == GPIOC)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
    if (GPIOx == GPIOD)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD, ENABLE);
    if (GPIOx == GPIOE)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE, ENABLE);
    if (GPIOx == GPIOF)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF, ENABLE);
    if (GPIOx == GPIOG)  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOF, ENABLE);

    DHT11_Mode_Out_PP();
    DHT11_BUS_HIGH;
}


// 从DHT11读取一个字节,MSB先行
// 1:每bit以50us低电平标置开始,
// 2:以26~28us的高电平表示“0”
// 3:以70us高电平表示“1”
// 4: 通过检测从高电平开后 x us后的电平即可区别这两个状态
static uint8_t readByte(void)
{
    uint8_t  temp = 0;

    for (uint8_t i = 0; i < 8; i++)
    {
        while (DHT11_BUS_READ == 0);             // 每bit以50us低电平标置开始,轮询直到从机发出 的50us 低电平 结束
        delay_us(40);                             // 延时x us 这个延时需要大于数据0持续的时间即可
        temp <<= 1;
        if (DHT11_BUS_READ == 1)                 // x us后仍为高电平表示数据“1”
        {
            while (DHT11_BUS_READ == Bit_SET);  // 等待数据1的高电平结束
            temp |= 1;                           // 位置“1“
        }
    }
    return temp;
}


/******************************************************************************
 * 函  数: DHT11_GetData
 * 功  能: 从DHT11中读取数据
 * 参  数: 1:
 * 说  明: 调用后,获取到的数据,保存到结构体xDHT11中
 *          温度值:xDHT11.Temperature (有效范围:0~50℃)
 *          湿度值: xDHT11.Humidity    (有效范围:20%~90%)
 * 返回值: 0-失败; 1-正常
 ******************************************************************************/
uint8_t DHT11_GetData(GPIO_TypeDef *GPIOx, uint32_t PINx)
{
    static uint8_t humiInt = 0;        // 湿度的整数部分
    static uint8_t humiDec = 0;        // 湿度的小数部分
    static uint8_t TempInt = 0;        // 温度的整数部分
    static uint8_t TempDec = 0;        // 温度的小数部分
    static uint8_t sum = 0;            // 校验和

    DHT11_Init(GPIOx, PINx);

    // 1:主机产生开始信号
    DHT11_Mode_Out_PP();               // 输出模式
    DHT11_BUS_LOW;                     // 主机拉低电平
    delay_ms(25);                      // 延时18~30ms
    // 2:主机拉高等待
    DHT11_BUS_HIGH;                    // 总线拉高
    delay_us(50);                      // 延时20~40us,这里设置50,是因为要直接进入下一个时序(电平状态),以方便检测

    // 3: 从机产生响应和准备信号
    DHT11_Mode_IPU();                  // 主机设为输入 判断从机响应信号

    if (DHT11_BUS_READ == 0)           // 判断从机是否产生响应信号_低电平, 如不响应则跳出
    {
        while (DHT11_BUS_READ == 0);   // 等待响应信号结束:低电平持续约80us
        while (DHT11_BUS_READ == 1);   // 等待标置信号结束:高电平持续约80us
        // 4: 从机连续输出5字节数据
        humiInt = readByte();          // 湿度的整数部分// 开始接收数据
        humiDec = readByte();          // 湿度的小数部分
        TempInt = readByte();          // 温度的整数部分
        TempDec = readByte();          // 温度的小数部分
        sum     = readByte();          // 校验和

        DHT11_Mode_Out_PP();           // 读取结束,引脚改为输出模式
        DHT11_BUS_HIGH;                // 主机拉高

        // 5: 检查读取的数据是否正确
        if (sum == (humiInt + humiDec + TempInt + TempDec))
        {
            xDHT11.Humidity = humiInt + humiDec;
            xDHT11.Temperature = (float)TempInt  + (float)TempDec / 10;
            return SUCCESS;            // 校检正确, 返回:SUCCESS=1
        }
        return ERROR;                  // 校检错误,返回:ERROR=0
    }
    return ERROR;                      // 通信错误,返回:ERROR=0
}




/******************************************************************************
 * 函  数: DHT11_GetTemperature
 * 功  能: 从DHT11中读取温度值
 * 参  数: GPIO_TypeDef *GPIOx: GPIO端口号,取值范围:GPIOA ~ GPIOG
 *          uint32_t     PINx  : 引脚编号,  取值范围:GPIO_Pin_0 ~ GPIO_Pin_15
 * 说  明: 温度值有效范围:0~50℃; 精度±2°C; 小数部份无效
 * 返回值: 0-失败,非0值-湿度值
 ******************************************************************************/
float DHT11_GetTemperature(GPIO_TypeDef *GPIOx, uint32_t PINx)
{ 
    DHT11_GetData(GPIOx, PINx);
    return xDHT11.Temperature;
}


/******************************************************************************
 * 函  数: DHT11_GetHumidity
 * 功  能: 从DHT11中读取湿度值
 * 参  数: GPIO_TypeDef *GPIOx: GPIO端口号,取值范围:GPIOA ~ GPIOG
 *          uint32_t     PINx  : 引脚编号,  取值范围:GPIO_Pin_0 ~ GPIO_Pin_15
 * 说  明: 湿度值有效范围:20%~90%; 精度±5%; 小数部分无效
 * 返回值: 0-失败,非0值-湿度值
 ******************************************************************************/
float DHT11_GetHumidity(GPIO_TypeDef *GPIOx, uint32_t PINx)
{  
    DHT11_GetData(GPIOx, PINx);
    return xDHT11.Humidity;
}

  • 🌿bsp_DHT11.h
#ifndef __BSP_DHT11_H
#define __BSP_DHT11_H
/***********************************************************************************************************************************
 **【文件名称】  bsp_DHT11.h
 **【功能测试】  DHT11-温湿度获取
 **【文件名称】  bsp_DHT11.h
 **【功能描述】  定义引脚、定义全局结构体、声明全局函数
 **
 **【适用平台】  STM32F103 + 标准库v3.5 + keil5
 **
 **【实验操作】  1-模块接线,VCC  接 3.3V 或 5V
 **                          DATA 接 PC3
 **                          GND  接 GND
 **
 **【划 重 点】  1-电压范围:3.3V~5.0V,工作电流:0.2~1mA; 待机电流:150uA
 **              2-DHT11反应有点慢:上电后1S后进入稳定状态; 
 **              3-采样周期,不同店家的DHT11,参数不同,一般是1次/秒, 有部分可以数百毫秒间隔采集一次; 当小于采样周期进行重复采集数据,会发生错误;
 **              5-DHT11时序要求较高,us级别,当使用delay粗略延时,同一份代码同一个模块,在不同keil里跑也可能发生错误(代码优化设置对while延时的影响)。建议使用TIM的精准延时;
 **              4-注意判断使用的DHT11器件,是单独的DHT11元件,还是完整的DHT11模块
 **                单独的DHT11元件:蓝色塑料主体、三个引脚,不带上拉电阻;适合低成本工程方案使用,建议测试时引脚使用推挽输出模式;
 **                完整的DTH11模块:蓝色塑料主体、三个引脚、PCB底板、上拉电阻、电源指示LED,适合方案搭建;
 **              6-精度误差:DHT11分旧版和新版
 **                          旧版数据稳定,但只有整数部分,新版有小数部分,但数据跳动稍大;
 **                          温度0-50°C; 精度±2°C; 小数部份无效
 **                          湿度20-80%; 精度±5%; 小数部分无效
 **
 **
 **【文件移植】  步骤1-复制文件:可复制bsp_DHT11.c和bsp_DHT11h两个文件,或复制DHT11文件夹,保存到所需工程目录文件夹下;
 **              步骤2-添加文件:在keil工各程左侧文件管理器中,双击某文件夹,以添加bsp_DHT11.c文件;
 **              步骤3-添加路径:点击魔术棒工具按钮,在“c/c++"选项页中,点击”Include Path"后面的按键,以添加文件存放所在路径(是文件夹,不是文件);
 **         步骤4-添加引脚:在所需DHT11功能的代码文件头部,添加:#include "bsp_DHT11.h";
 **
 **
 **【函数使用】  函数2-DHT11_GetData(GPIOx, GPIO_Pin_x);           // 获取数据, 获取的数据存放于结构体xDHT11中,具体使用方法,可参考示例代码
 **
***********************************************************************************************************************************/
#include <stm32f10x.h>
#include <stdio.h>


#define   DELAY_MODE_TIM2   1     // 通信时序中的延时方式:0_使用while方式的粗略延时; 1_使用TIM2进行计时; 注意计时器是否和其它设备有冲突
                                  // 0_使用粗略延时:注意本代码是在勾选下列参数状态下调试的:Options / C++ /One ELF Section per Function ;
                                  // 1_使用TIM2计时:首选推荐; 注意同一工程中是否有其它功能使用TIM2,以免发生冲突;

/*****************************************************************************
 ** 声明 全局变量
****************************************************************************/
typedef struct
{
    float          Temperature;   // 温度值:在调用DHT11_GetTemp()函数后,获取到的温度值;
    float          Humidity;      // 湿度值:在调用DHT11_GetTemp()函数后,获取到的温度值;
} xDHT11_TypeDef;
extern xDHT11_TypeDef  xDHT11;    // 声明全局结构体, 用于存放读取的结果值



/*****************************************************************************
 ** 声明  全局函数
****************************************************************************/
uint8_t DHT11_GetData(GPIO_TypeDef *GPIOx, uint32_t PINx);
float DHT11_GetTemperature(GPIO_TypeDef *GPIOx, uint32_t PINx);
float DHT11_GetHumidity(GPIO_TypeDef *GPIOx, uint32_t PINx);

#endif 

  • 📝main.c
#include <stm32f10x.h>            // 头文件引用(标准库); 内核、芯片外设....;(stm32f10x.conf.h, 对标准库头文件进行调用)     
#include "stm32f10x_conf.h"       // 头文件引用(标准库); 内核、芯片外设....;(stm32f10x.conf.h, 对标准库头文件进行调用) 
#include "bsp_led.h"              // LED指示灯
#include "bsp_usart.h"            // USART1、2、3,UART4、5
#include "bsp_lcd_ST7735.h"       // 1.8寸显示屏驱动 
//#include "bsp_w25Qxx.h"           // 外部FLASH,用于中文支持
#include "bsp_DHT11.h"            // DHT11

float temperature = 0.0;          // 用于存储温度值; DHT11的精度,小数部分无效
float humidity = 0.0;             // 用于存储湿度值; DHT11的精度,小数部分无效
char   strTem[12];               // 用于临时存储字符串


// ms延时函数,减少移植时对外部文件依赖;
static void delay_ms(u32 ms)
{
    ms = ms * 6500;
    for (u32 i = 0; i < ms; i++); // 72MHz系统时钟下,多少个空循环约耗时1ms
}


// 主函数
int main(void)
{
    USART1_Init(115200);                                        // 串口初始化:USART1(115200-N-8-1), 且工程已把printf重定向至USART1输出

    Led_Init();                                                 // LED 初始化
    LED_RED_ON;                                                 // 点亮红灯

//    W25qx_Init();                                               // 初始化外部Flash, 已烧录有中文字库

    LCD_Init();                                                 // 初始化显示屏-驱动芯片ST7735
    LCD_String(25, 3, "DHT11 DATA", 16, WHITE, BLACK);      // 预先显示固定的屏显内容,不用重复刷新占用芯片资源
    LCD_Line(0, 25, 127, 25, WHITE);
    LCD_String(5, 35, "Temp Value:", 12, WHITE, BLACK);
    LCD_String(5, 55, "Humi Value:", 12, WHITE, BLACK);
		LCD_Chinese16ForFile(101, 35, 6, GREEN, BLACK);
    while (1)                                                   // while函数死循环,不能让main函数运行结束,否则会产生硬件错误
    {
        delay_ms(500);                                          // 延时500ms
        LED_RED_TOGGLE;                                         // 红色LED,间隔亮灭,以判断系统正常工作

        printf("DHT11  ");

        temperature = DHT11_GetTemperature(GPIOC, GPIO_Pin_3);  // 获取DHT11温度值, 注意,DHT11的精度,小数部分无效
        printf("温度:%4.1f℃   ", temperature);                  // 把数据输出到电脑串口软件,方便观察
        sprintf(strTem, "%4.1f", temperature);              // 格式化,把float值存储为字符数组
        LCD_String(75, 35, strTem, 12, GREEN, BLACK);           // 温度值显示在显示屏上

        humidity =  DHT11_GetHumidity(GPIOC, GPIO_Pin_3);       // 获取DHT11湿度值, 注意,DHT11的精度,小数部分无效
        printf("湿度:%4.1fRH%%\r\n", humidity);                 // 把数据输出到电脑串口软件,方便观察
        sprintf(strTem, "%4.1f%%  ", humidity);                 // 格式化,把float值存储为字符数组
        LCD_String(75, 55, strTem, 12, GREEN, BLACK);           // 温度值显示在显示屏上
    }
}

📚工程源码

链接:https://pan.baidu.com/s/1k6iZADRF0rSesPrTYK7rxA 
提取码:82cz
  • 0
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值