STM32F103基于I2C协议的AHT20温湿度传感器的数据采集

一、I2C原理

软件I2C和硬件I2C

软件I2C:用单片机的两个I/O端口模拟出来的I2C,用软件控制管脚状态以模拟I2C通信波形,软件模拟寄存器的工作方式。

直接使用 CPU 内核按照 I2C 协议的要求控制 GPIO 输出高低电平,从而模拟I2C。
使用: 需要在控制产生 I2C 的起始信号时,控制作为SCL 线的 GPIO 引脚输出高电平,然后控制作为 SDA 线的 GPIO 引脚在此期间完成由高电平至低电平的切换,最后再控制SCL线切换为低电平,这样就输出了一个标准的 I2C 起始信号。

硬件I2C:一块硬件电路,硬件I2C对应芯片上的I2C外设,有相应I2C驱动电路,其所使用的I2C管脚也是专用的,硬件(固件)I2C是直接调用内部寄存器进行配置。

直接利用 STM32 芯片中的硬件 I2C 外设。
使用: 只要配置好对应的寄存器,外设就会产生标准串口协议的时序。在初始化好 I2C 外设后,只需要把某寄存器位置 1,此时外设就会控制对应的 SCL 及 SDA 线自动产生 I2C 起始信号,不需要内核直接控制引脚的电平。

硬件I2C的效率要远高于软件的,而软件I2C由于不受管脚限制,接口比较灵活。

如何区分:

可以看底层配置,比如IO口配置,如果配置了IO口的功能(I2C功能)那就是固件I2C,否则就是模拟
可以看I2C写函数,看里面有木有调用现成的函数或者给某个寄存器赋值,如果有,则肯定是固件I2C功能,没有的话肯定是数据一个bit一个bit模拟发生送的,肯定用到了循环,则为模拟。
根据代码量判断,模拟的代码量肯定比固件的要大。

硬件I2C用法比较复杂,模拟I2C的流程更清楚一些。

硬件I2C速度比模拟快,并且可以用DMA

模拟I2C可以在任何管脚上,而硬件只能在固定管脚上。

软件i2c是程序员使用程序控制SCL,SDA线输出高低电平,模拟i2c协议的时序。一般较硬件i2c稳定,但是程序较为繁琐,但不难。

硬件i2c程序员只要调用i2c的控制函数即可,不用直接的去控制SCL,SDA高低电平的输出。但是有些单片机的硬件i2c不太稳定,调试问题较多。
参考:https://blog.csdn.net/weixin_52650112/article/details/127688834

二、[STM32F103]基于I2C协议的AHT20温湿度采集

1.配置引脚以及串口

在这里插入图片描述

2.主要代码

main.c
/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * @file           : main.c
  * @brief          : Main program body
  ******************************************************************************
  * @attention
  *
  * <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
  * All rights reserved.</center></h2>
  *
  * This software component is licensed by ST under BSD 3-Clause license,
  * the "License"; You may not use this file except in compliance with the
  * License. You may obtain a copy of the License at:
  *                        opensource.org/licenses/BSD-3-Clause
  *
  ******************************************************************************
  */
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "dma.h"
#include "i2c.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */

#include<stdio.h>
#include "AHT20-21_DEMO_V1_3.h" 

void SystemClock_Config(void);


int fputc(int ch,FILE *f)
 
{
    HAL_UART_Transmit(&huart1,(uint8_t *)&ch,1,0xFFFF);    
		//等待发送结束	
		while(__HAL_UART_GET_FLAG(&huart1,UART_FLAG_TC)!=SET){
		}		

    return ch;
}



int main(void)
{
  /* USER CODE BEGIN 1 */
	uint32_t CT_data[2]={0,0};
	volatile int  c1,t1;
	
	Delay_1ms(500);

	HAL_Init();

	SystemClock_Config();

	MX_GPIO_Init();
	MX_DMA_Init();
	MX_USART1_UART_Init();
	
	//初始化AHT20
	AHT20_Init();
	Delay_1ms(500);

  while (1)
  { 
    /* USER CODE END WHILE */
		AHT20_Read_CTdata(CT_data);       //不经过CRC校验,直接读取AHT20的温度和湿度数据    推荐每隔大于1S读一次
		//AHT20_Read_CTdata_crc(CT_data);  //crc校验后,读取AHT20的温度和湿度数据 
	

		c1 = CT_data[0]*1000/1024/1024;  //计算得到湿度值c1(放大了10倍)
		t1 = CT_data[1]*2000/1024/1024-500;//计算得到温度值t1(放大了10倍)
		printf("正在检测");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		printf("\r\n");
		HAL_Delay(1000);
		printf("温度:%d%d.%d",t1/100,(t1/10)%10,t1%10);
		printf("湿度:%d%d.%d",c1/100,(c1/10)%10,c1%10);
		printf("\r\n");
		printf("等待");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		HAL_Delay(100);
		printf(".");
		printf("\r\n");
		HAL_Delay(1000);
  /* USER CODE END 3 */
	}
}

/**
  * @brief System Clock Configuration
  * @retval None
  */
void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
  /** Initializes the CPU, AHB and APB buses clocks
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK)
  {
    Error_Handler();
  }
}

/* USER CODE BEGIN 4 */

/* USER CODE END 4 */

/**
  * @brief  This function is executed in case of error occurrence.
  * @retval None
  */
void Error_Handler(void)
{
  /* USER CODE BEGIN Error_Handler_Debug */
  /* User can add his own implementation to report the HAL error return state */
  __disable_irq();
  while (1)
  {
  }
  /* USER CODE END Error_Handler_Debug */
}

#ifdef  USE_FULL_ASSERT
/**
  * @brief  Reports the name of the source file and the source line number
  *         where the assert_param error has occurred.
  * @param  file: pointer to the source file name
  * @param  line: assert_param error line source number
  * @retval None
  */
void assert_failed(uint8_t *file, uint32_t line)
{
  /* USER CODE BEGIN 6 */
  /* User can add his own implementation to report the file name and line number,
     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  /* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */

/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/

3.烧录程序

在这里插入图片描述
硬件连接:
在这里插入图片描述
STM32F103C8T6——USB to TTL
3V3 ——3V3
GND —— GND
RXD ——A9
TXD ——A10

STM32F103C8T6 ——AHT20
3V3/5V——VDD
PB7——SDA
GND——GND
PB6——SCL
参考:https://blog.csdn.net/weixin_52650112/article/details/127688834

4.结果展示:

在这里插入图片描述

(这里没解决好文字部分是乱码,意思已经标注好)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 我可以给你一些建议:1.使用STM32CubeMX可以快速生成STM32F103C8T6的初始化代码;2.使用I2C库函数来完成AHT21与STM32F103C8T6之间的通信;3.根据AHT21的数据手册,编写读取湿度数据的代码;4.调试代码确保湿度读取能够正常工作。 ### 回答2: STM32F103C8T6是一款来自STMicroelectronics的微控制器,它采用了ARM Cortex-M3内核,具有丰富的外设和强大的性能。AHT21是一款高精度、低功耗的湿度传感器。 要完成STM32F103C8T6基于AHT21的湿度读取程序,我们可以按照以下步骤进行: 1. 确定硬件连接:将AHT21传感器的引脚与STM32F103C8T6的GPIO引脚相连接。一般来说,AHT21的SCL引脚连接到STM32的SCL引脚,而SDA引脚连接到STM32的SDA引脚。 2. 确定软件开发环境:选择合适的集成开发环境(IDE)进行编程,如Keil MDK、STM32CubeIDE等。 3. 创建STM32程序:在IDE中创建一个新的STM32项目。 4. 配置I2C总线:在STM32I2C外设配置中,设置适当的时钟频率和其他相关参数。确保I2C总线与AHT21的通信速率兼容。 5. 编写读取程序:在主程序代码中,使用适当的I2C读取函数来读取AHT21的湿度数据。具体的读取流程可以参考AHT21的数据手册,其中包含了读取湿度数据的详细指令描述。 6. 处理数据:将读取到的湿度原始数据进行适当的计算和转换,以得到实际的湿度值。一般来说,AHT21的数据是以二进制格式进行传输的,需要进行一些转换和校准才能得到最终的湿度值。 7. 输出数据:可以选择以串口、LCD显示屏等方式输出湿度数据,便于用户查看。 在完成以上步骤后,将编译、烧录程序到STM32F103C8T6微控制器,并将AHT21连接至对应引脚后,程序就可以读取AHT21传感器的湿度数据了。 ### 回答3: 要基于STM32F103C8T6完成AHT21湿度读取程序,首先需要了解AHT21传感器的工作原理和STM32F103C8T6的开发环境。 AHT21湿度传感器是一种数字式湿度传感器,采用I2C总线进行通信。它具有高精度、低功耗和快速响应的特点。 STM32F103C8T6是一款基于ARM Cortex-M3内核的低功耗微控制器。它具有丰富的外设和强大的计算能力,适用于各种嵌入式应用。 下面是基于STM32F103C8T6完成AHT21湿度读取程序的步骤: 1. 硬件连接:将AHT21传感器的SDA引脚连接到STM32F103C8T6的I2C1的SDA引脚,将SCL引脚连接到I2C1的SCL引脚。同时,将AHT21的VCC引脚连接到3.3V电源,将GND引脚连接到地。 2. 初始化I2C1外设:在STM32F103C8T6的开发环境中,使用HAL库可以方便地初始化和配置I2C1外设。需要配置I2C1的时钟频率、引脚以及其他参数。 3. 发送读取命令:使用I2C1发送读取AHT21度和湿度的命令。命令为0xAC,将其发送到AHT21传感器。 4. 接收数据:等待AHT21传感器返回数据,使用I2C1接收数据,并将其保存到缓冲区中。 5. 数据处理:从接收到的数据中提取度和湿度的数值,并进行必要的处理,例如转换为实际度和湿度值。 6. 显示结果:将湿度值显示到相应的显示设备,例如OLED屏幕或串口。 以上就是基于STM32F103C8T6完成AHT21湿度读取程序的大致步骤。具体实现可以根据具体需求和开发环境做相应的调整。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值