STM32F103驱动SHT20温湿度传感器代码实现

基于STM32F103芯片驱动SHT20温湿度传感器的实现。SHT20是一款高精度、低功耗的温湿度传感器,通过I2C接口与微控制器通信。

硬件连接

首先,确保SHT20与STM32F103的正确连接:

  • SHT20的VDD引脚 -> 3.3V
  • SHT20的GND引脚 -> GND
  • SHT20的SCL引脚 -> STM32的PB6(I2C1_SCL)或PB10(I2C2_SCL)
  • SHT20的SDA引脚 -> STM32的PB7(I2C1_SDA)或PB11(I2C2_SDA)

代码实现

1. 头文件定义 (sht20.h)

#ifndef __SHT20_H
#define __SHT20_H

#include "stm32f10x.h"
#include "delay.h"

// SHT20 I2C地址
#define SHT20_ADDRESS 0x40

// SHT20命令定义
#define SHT20_TRIG_T_MEASUREMENT_HOLD  0xE3
#define SHT20_TRIG_RH_MEASUREMENT_HOLD 0xE5
#define SHT20_TRIG_T_MEASUREMENT_NO_HOLD 0xF3
#define SHT20_TRIG_RH_MEASUREMENT_NO_HOLD 0xF5
#define SHT20_WRITE_USER_REG           0xE6
#define SHT20_READ_USER_REG            0xE7
#define SHT20_SOFT_RESET               0xFE

// 用户寄存器位定义
#define SHT20_USER_REG_RESOLUTION_RH12_T14 0x00
#define SHT20_USER_REG_RESOLUTION_RH8_T12  0x01
#define SHT20_USER_REG_RESOLUTION_RH10_T13 0x80
#define SHT20_USER_REG_RESOLUTION_RH11_T11 0x81
#define SHT20_USER_REG_END_OF_BATTERY      0x40
#define SHT20_USER_REG_HEATER_ENABLE       0x04
#define SHT20_USER_REG_OTP_RELOAD_DISABLE  0x02

// 函数声明
void SHT20_Init(void);
void SHT20_Reset(void);
uint8_t SHT20_ReadUserRegister(void);
void SHT20_WriteUserRegister(uint8_t reg);
float SHT20_ReadTemperature(void);
float SHT20_ReadHumidity(void);
uint16_t SHT20_ReadRawValue(uint8_t command);
uint8_t SHT20_CheckCRC(uint16_t data, uint8_t crc);

#endif /* __SHT20_H */

2. 源文件实现 (sht20.c)

#include "sht20.h"
#include "i2c.h"  // 需要包含你的I2C驱动头文件

// 初始化SHT20
void SHT20_Init(void)
{
    // 初始化I2C
    I2C_Init(); // 确保你已经实现了I2C初始化函数
    
    // 软复位SHT20
    SHT20_Reset();
    
    // 延时等待复位完成
    Delay_ms(15);
    
    // 设置分辨率为RH12-T14(默认值)
    SHT20_WriteUserRegister(SHT20_USER_REG_RESOLUTION_RH12_T14);
}

// 软复位SHT20
void SHT20_Reset(void)
{
    I2C_Start();
    I2C_SendByte(SHT20_ADDRESS << 1); // 写模式
    I2C_WaitAck();
    I2C_SendByte(SHT20_SOFT_RESET);
    I2C_WaitAck();
    I2C_Stop();
}

// 读取用户寄存器
uint8_t SHT20_ReadUserRegister(void)
{
    uint8_t reg = 0;
    
    I2C_Start();
    I2C_SendByte(SHT20_ADDRESS << 1); // 写模式
    I2C_WaitAck();
    I2C_SendByte(SHT20_READ_USER_REG);
    I2C_WaitAck();
    
    I2C_Start();
    I2C_SendByte((SHT20_ADDRESS << 1) | 0x01); // 读模式
    I2C_WaitAck();
    reg = I2C_ReadByte();
    I2C_NAck();
    I2C_Stop();
    
    return reg;
}

// 写入用户寄存器
void SHT20_WriteUserRegister(uint8_t reg)
{
    I2C_Start();
    I2C_SendByte(SHT20_ADDRESS << 1); // 写模式
    I2C_WaitAck();
    I2C_SendByte(SHT20_WRITE_USER_REG);
    I2C_WaitAck();
    I2C_SendByte(reg);
    I2C_WaitAck();
    I2C_Stop();
}

// 读取原始数据
uint16_t SHT20_ReadRawValue(uint8_t command)
{
    uint8_t data[3] = {0};
    uint16_t raw_value = 0;
    
    I2C_Start();
    I2C_SendByte(SHT20_ADDRESS << 1); // 写模式
    I2C_WaitAck();
    I2C_SendByte(command);
    I2C_WaitAck();
    
    // 如果是无保持模式,需要等待测量完成
    if(command == SHT20_TRIG_T_MEASUREMENT_NO_HOLD || 
       command == SHT20_TRIG_RH_MEASUREMENT_NO_HOLD)
    {
        // 等待测量完成,SDA线会被拉低
        I2C_Start();
        while(I2C_SendByte((SHT20_ADDRESS << 1) | 0x01) != 0);
    }
    else
    {
        // 保持模式下直接读取
        Delay_ms(100); // 根据分辨率不同,等待时间不同
        
        I2C_Start();
        I2C_SendByte((SHT20_ADDRESS << 1) | 0x01); // 读模式
        I2C_WaitAck();
    }
    
    // 读取3个字节:2个数据字节 + 1个CRC字节
    data[0] = I2C_ReadByte();
    I2C_Ack();
    data[1] = I2C_ReadByte();
    I2C_Ack();
    data[2] = I2C_ReadByte();
    I2C_NAck();
    I2C_Stop();
    
    // 检查CRC
    raw_value = (data[0] << 8) | data[1];
    if(SHT20_CheckCRC(raw_value, data[2]))
    {
        return raw_value;
    }
    else
    {
        return 0xFFFF; // CRC错误
    }
}

// CRC校验
uint8_t SHT20_CheckCRC(uint16_t data, uint8_t crc)
{
    uint32_t remainder = (uint32_t)data << 8;
    uint32_t divsor = (uint32_t)0x988000; // CRC多项式: x^8 + x^5 + x^4 + 1 (0x131)
    
    remainder |= crc;
    
    for(int i = 0; i < 16; i++)
    {
        if(remainder & (uint32_t)1 << (23 - i))
        {
            remainder ^= divsor;
        }
        divsor >>= 1;
    }
    
    return (remainder == 0);
}

// 读取温度值
float SHT20_ReadTemperature(void)
{
    uint16_t raw_temp = SHT20_ReadRawValue(SHT20_TRIG_T_MEASUREMENT_HOLD);
    
    if(raw_temp == 0xFFFF)
    {
        return -273.15f; // 错误值
    }
    
    // 转换为实际温度值
    return -46.85f + 175.72f * (raw_temp / 65536.0f);
}

// 读取湿度值
float SHT20_ReadHumidity(void)
{
    uint16_t raw_humidity = SHT20_ReadRawValue(SHT20_TRIG_RH_MEASUREMENT_HOLD);
    
    if(raw_humidity == 0xFFFF)
    {
        return -1.0f; // 错误值
    }
    
    // 转换为实际湿度值
    return -6.0f + 125.0f * (raw_humidity / 65536.0f);
}

3. I2C驱动示例 (i2c.c)

这里提供一个简单的I2C驱动实现,你需要根据你的硬件配置进行调整:

#include "i2c.h"
#include "stm32f10x.h"

// I2C初始化
void I2C_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    I2C_InitTypeDef I2C_InitStructure;
    
    // 使能I2C和GPIO时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE);
    
    // 配置I2C引脚: SCL-PB6, SDA-PB7
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD; // 开漏输出
    GPIO_Init(GPIOB, &GPIO_InitStructure);
    
    // I2C配置
    I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;
    I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;
    I2C_InitStructure.I2C_OwnAddress1 = 0x00;
    I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;
    I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;
    I2C_InitStructure.I2C_ClockSpeed = 400000; // 400kHz
    
    I2C_Init(I2C1, &I2C_InitStructure);
    I2C_Cmd(I2C1, ENABLE);
}

// 产生I2C起始信号
void I2C_Start(void)
{
    I2C_GenerateSTART(I2C1, ENABLE);
    while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_MODE_SELECT));
}

// 产生I2C停止信号
void I2C_Stop(void)
{
    I2C_GenerateSTOP(I2C1, ENABLE);
}

// 发送一个字节
void I2C_SendByte(uint8_t data)
{
    I2C_SendData(I2C1, data);
    while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
}

// 读取一个字节
uint8_t I2C_ReadByte(void)
{
    while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_RECEIVED));
    return I2C_ReceiveData(I2C1);
}

// 等待应答
uint8_t I2C_WaitAck(void)
{
    while(!I2C_CheckEvent(I2C1, I2C_EVENT_MASTER_BYTE_TRANSMITTED));
    return I2C_GetFlagStatus(I2C1, I2C_FLAG_AF) == RESET;
}

// 发送应答
void I2C_Ack(void)
{
    I2C_AcknowledgeConfig(I2C1, ENABLE);
}

// 发送非应答
void I2C_NAck(void)
{
    I2C_AcknowledgeConfig(I2C1, DISABLE);
}

4. 主程序示例 (main.c)

#include "stm32f10x.h"
#include "sht20.h"
#include "stdio.h"  // 用于printf

// 重定向printf到串口
int fputc(int ch, FILE *f)
{
    USART_SendData(USART1, (uint8_t)ch);
    while(USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET);
    return ch;
}

// 初始化串口
void USART1_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;
    
    // 使能时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA, ENABLE);
    
    // 配置USART1 Tx (PA9)为推挽复用输出
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    
    // 配置USART1 Rx (PA10)为浮空输入
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
    
    // USART参数配置
    USART_InitStructure.USART_BaudRate = 115200;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
    
    USART_Init(USART1, &USART_InitStructure);
    USART_Cmd(USART1, ENABLE);
}

int main(void)
{
    float temperature, humidity;
    
    // 初始化系统时钟
    SystemInit();
    
    // 初始化串口
    USART1_Init();
    
    // 初始化SHT20
    SHT20_Init();
    
    printf("SHT20 Temperature & Humidity Sensor Test\r\n");
    
    while(1)
    {
        // 读取温湿度
        temperature = SHT20_ReadTemperature();
        humidity = SHT20_ReadHumidity();
        
        // 打印结果
        printf("Temperature: %.2f C, Humidity: %.2f %%RH\r\n", temperature, humidity);
        
        // 延时2秒
        Delay_ms(2000);
    }
}

参考代码 基于STM32F103芯片的,温湿度传感器SHT20 www.youwenfan.com/contentcne/103320.html

说明

  1. 将上述代码添加到你的STM32项目中
  2. 根据你的硬件配置调整I2C引脚和时钟设置
  3. 确保已正确实现延时函数(Delay_ms)
  4. 编译并下载程序到STM32F103
  5. 通过串口监视器查看温湿度数据
posted @ 2025-08-28 12:02  吴逸杨  阅读(244)  评论(0)    收藏  举报