【STM32入门学习】基于DHT20温湿度传感器数据采集和显示

目录

一、软件I2C和硬件I2C

1.1硬件I2C

1.2软件I2C

二、创建工程(基于HAL库)

三、实物连线

3.1DHT20

3.2串口传输:

四、结果演示

4.1部署文件

4.2主要代码分析

五、心得体会


一、软件I2C和硬件I2C

I2C 基本读写过程:

其中S表示由主机的I2C接口产生的传输起始信号(S),这时连接到I2C总线上的所有从机都会接收到这个信号。

起始信号产生后,所有从机就开始等待主机紧接下来广播 的从机地址信号 (SLAVE_ADDRESS)。在I2C总线上, 每个设备的地址都是唯一的,当主机广播的地址与某个设备地址相同时,这个设备就被选中了,没被选中的设备将会忽略之后的数据信号。 根据I2C协议,这个从机地址可以是7位或10位。

在地址位之后,是传输方向的选择位,该位为0时,表示后面的数据传输方向是由主机传输至从机,即主机向从机写数据。该位为1时,则相反,即主机由从机读数据。

从机接收到匹配的地址后,主机或从机会返回一个应答(ACK)或非应答(NACK)信号,只有接收到应答信号后,主机才能继续发送或接收数据。

1.1硬件I2C

"软件I2C"和"硬件I2C"是两种不同的实现方式,用于在微控制器或其他嵌入式系统中控制I2C(Inter-Integrated Circuit)设备(如传感器、显示屏等)的通信。它们有着各自的特点和适用场景:

硬件I2C指的是利用微控制器芯片内置的硬件模块来实现I2C通信。这种方法通常具有以下特点:

1.速度和精确性: 硬件I2C利用专门的硬件电路实现,能够以较高的速度进行通信,并且通常具有更高的时序精确度,有助于在高速通信时避免时序问题。

2.低占用率: 使用硬件模块进行I2C通信可以减少微控制器的处理负担,因为大部分的通信和时序控制由硬件完成,减少了CPU的介入和处理时间。

3.可靠性高: 硬件I2C通常能够更可靠地处理I2C总线上的冲突和错误情况,因为硬件模块设计用来处理这些情况。

4.电路设计简单: 在电路设计时,硬件I2C只需连接到微控制器的特定引脚,并设置相应的寄存器配置即可使用。

1.2软件I2C

软件I2C是通过软件实现的I2C协议栈,通常是在没有硬件I2C模块或需要额外的I2C接口时使用。它具有以下特点:

适用性广: 可以在几乎所有的微控制器上实现,即使微控制器本身没有硬件I2C模块,也可以通过软件模拟来实现I2C通信。

灵活性: 软件I2C可以根据具体应用需求进行调整和优化,例如改变时序、处理特定的I2C设备或特定的通信速率。

资源占用高: 由于是通过软件来模拟I2C协议,因此会占用微控制器的处理资源,特别是在高速通信或复杂的通信场景下,可能会引入额外的延迟或稳定性问题。

不适合高速通信: 由于受限于软件处理能力和时序控制的精度,软件I2C通常不适合用于高速I2C通信,因为无法达到硬件I2C的速度和稳定性。

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

而硬件I2C是指直接利用STM32芯片中的硬件I2C外设,该硬件I2C外设跟USART串口外设类似,只要配置好对应的寄存器, 外设就会产生标准串口协议的时序。使用它的I2C外设则可以方便地通过外设寄存器产生I2C协议方式的通讯,如初始化好I2C外设后, 只需要把某寄存器位置1,那么外设就会控制对应的SCL及SDA线自动产生I2C起始信号,而不需要内核直接控制引脚的电平。

相对来说,硬件I2C直接使用外设来控制引脚,可以减轻CPU的负担。不过使用硬件I2C时必须使用某些固定的引脚作为SCL和SDA, 软件模拟I2C则可以使用任意GPIO引脚,相对比较灵活。在本开发板中,由于STM32RCT6芯片引脚较少,资源比较紧张, 在设计硬件时不方便使用硬件I2C指定的引脚连接外部设备(EEPROM存储器芯片),所以在控制程序上只能使用软件模拟I2C的方式。 若希望学习如何使用STM32的硬件I2C外设,可以参考我们其它开发板的教程,如F103指南者、F103霸道等开发板。

二、创建工程(基于HAL库)

我先放芯片最终配置的引脚,再一步一步介绍:

2.1HAL库创建程序:

我们可以根据创建好的芯片来找到我们的引脚配置:

三、实物连线

3.1DHT20

3.2串口传输:

串口TTL模块:

PA9的TX接TTL的RX

PA10RX接TTL的TX

四、结果演示

4.1部署文件

创建好文件准备充分后,需要把温湿度传感器代码嵌入并移植:

了解学习了I2C协议进行通信,需要对这个协议有比较清晰的了解。对DHT20芯片读取数据的过程也要有一个比较清晰的认识,才能看懂代码中是如何进行读取的。这次试验既是对I2C有了清晰的认识,也对C8T6有了更多的学习。可以打开已有的工程模板,也可以参考DHT20温度采集(http://t.csdnimg.cn/wx73o)中的代码部分,

或者由DHT20厂家提供的固件库撰写,厂家代码下载请点击百度网盘资料链接

https://pan.baidu.com/s/1OoHl2PPrb-RCFsbuaFhViA?pwd=8520下载提取码:8520

4.2主要代码分析

①DHT20芯片的使用过程

void  read_AHT20_once(void)
{
	delay_ms(10);

	reset_AHT20();//重置AHT20芯片
	delay_ms(10);

	init_AHT20();//初始化AHT20芯片
	delay_ms(10);

	startMeasure_AHT20();//开始测试AHT20芯片
	delay_ms(80);

	read_AHT20();//读取AHT20采集的到的数据
	delay_ms(5);
}

②AHT20芯片读取数据

void read_AHT20(void)
{
	uint8_t   i;
	for(i=0; i<6; i++)
	{
		readByte[i]=0;
	}
	I2C_Start();//I2C启动

	I2C_WriteByte(0x71);//I2C写数据
	ack_status = Receive_ACK();//收到的应答信息
	readByte[0]= I2C_ReadByte();//I2C读取数据
	Send_ACK();//发送应答信息

	readByte[1]= I2C_ReadByte();
	Send_ACK();

	readByte[2]= I2C_ReadByte();
	Send_ACK();

	readByte[3]= I2C_ReadByte();
	Send_ACK();

	readByte[4]= I2C_ReadByte();
	Send_ACK();

	readByte[5]= I2C_ReadByte();
	SendNot_Ack();
	//Send_ACK();

	I2C_Stop();//I2C停止函数
	//判断读取到的第一个字节是不是0x08,0x08是该芯片读取流程中规定的,如果读取过程没有问题,就对读到的数据进行相应的处理
	if( (readByte[0] & 0x68) == 0x08 )
	{
		H1 = readByte[1];
		H1 = (H1<<8) | readByte[2];
		H1 = (H1<<8) | readByte[3];
		H1 = H1>>4;

		H1 = (H1*1000)/1024/1024;

		T1 = readByte[3];
		T1 = T1 & 0x0000000F;
		T1 = (T1<<8) | readByte[4];
		T1 = (T1<<8) | readByte[5];

		T1 = (T1*2000)/1024/1024 - 500;

		AHT20_OutData[0] = (H1>>8) & 0x000000FF;
		AHT20_OutData[1] = H1 & 0x000000FF;

		AHT20_OutData[2] = (T1>>8) & 0x000000FF;
		AHT20_OutData[3] = T1 & 0x000000FF;
	}
	else
	{
		AHT20_OutData[0] = 0xFF;
		AHT20_OutData[1] = 0xFF;

		AHT20_OutData[2] = 0xFF;
		AHT20_OutData[3] = 0xFF;
		printf("读取失败!!!");

	}
	printf("\r\n");
	//根据AHT20芯片中,温度和湿度的计算公式,得到最终的结果,通过串口显示
	printf("温度:%d%d.%d",T1/100,(T1/10)%10,T1%10);
	printf("湿度:%d%d.%d",H1/100,(H1/10)%10,H1%10);
	printf("\r\n");
}

结果演示:

五、心得体会

本次实验实现了温湿度传感器数据读取并传输,对于STM32的学习来说,更进一步了,通过温湿度传感器,感知外界信息,实现单片机的交互,使得本人对于单片机的理解和掌握更进一步。当然,本人还有很多欠缺的地方,如果本文存在不妥之处,还望各位读者批评指正,感谢你的阅读。

参考文献:

【stm32】基于DHT20与I2C接口实现温湿度的采集_dht20sda浮空输入时数据全为0xff-CSDN博客

一文看懂I2C协议 - 知乎 (zhihu.com)

  • 18
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是基于STM32f103zet6的dht11湿度传感器数据采集代码,你可以根据自己的实际情况进行修改和调试: ```c #include "stm32f10x.h" #include "dht11.h" #include "delay.h" int main(void) { uint8_t res; uint8_t humi, temp; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能GPIOB时钟 DHT11_Init(); //初始化DHT11 while(1) { res = DHT11_Read_Data(&humi, &temp); //读取湿度数据 if(res == 0) { //读取成功 printf("Temperature: %d.%d *C, Humidity: %d.%d %%\r\n", temp/10, temp%10, humi/10, humi%10); } else { //读取失败 printf("Read error!\r\n"); } Delay_Ms(2000); //延时2s } } ``` 其中,`dht11.h`和`delay.h`是需要自己编写的头文件,可以参考下面的示例代码: ```c #ifndef __DHT11_H #define __DHT11_H #include "stm32f10x.h" #define DHT11_PIN GPIO_Pin_0 #define DHT11_GPIO GPIOB #define DHT11_RCC RCC_APB2Periph_GPIOB void DHT11_Init(void); uint8_t DHT11_Read_Byte(void); uint8_t DHT11_Read_Data(uint8_t *humi, uint8_t *temp); #endif ``` ```c #ifndef __DELAY_H #define __DELAY_H #include "stm32f10x.h" void Delay_Init(void); void Delay_Ms(uint16_t nms); void Delay_Us(uint32_t nus); #endif ``` 以下是`dht11.c`和`delay.c`的代码示例,你可以根据自己的需求进行修改和完善: ```c #include "dht11.h" #include "delay.h" //DHT11初始化 void DHT11_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(DHT11_RCC, ENABLE); //使能GPIOB时钟 GPIO_InitStructure.GPIO_Pin = DHT11_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //输出模式 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //50MHz GPIO_Init(DHT11_GPIO, &GPIO_InitStructure); //初始化GPIO } //DHT11读取一个字节 uint8_t DHT11_Read_Byte(void) { uint8_t i, byte = 0; for(i=0; i<8; i++) { while(!GPIO_ReadInputDataBit(DHT11_GPIO, DHT11_PIN)); //等待高电平 Delay_Us(30); //延时30us if(GPIO_ReadInputDataBit(DHT11_GPIO, DHT11_PIN)) { //如果仍是高电平 byte |= (1<<(7-i)); //写入数据,MSB先传输 while(GPIO_ReadInputDataBit(DHT11_GPIO, DHT11_PIN)); //等待低电平 } } return byte; } //DHT11读取湿度数据 uint8_t DHT11_Read_Data(uint8_t *humi, uint8_t *temp) { uint8_t buf[5], i; GPIO_ResetBits(DHT11_GPIO, DHT11_PIN); //发送起始信号 Delay_Ms(18); GPIO_SetBits(DHT11_GPIO, DHT11_PIN); Delay_Us(30); if(!GPIO_ReadInputDataBit(DHT11_GPIO, DHT11_PIN)) { //等待DHT11响应 while(!GPIO_ReadInputDataBit(DHT11_GPIO, DHT11_PIN)); //等待DHT11响应完成 for(i=0; i<5; i++) { buf[i] = DHT11_Read_Byte(); //读取5个字节的数据 } if((buf[0]+buf[1]+buf[2]+buf[3]) == buf[4]) { //校验和正确 *humi = buf[0]; *temp = buf[2]; return 0; //返回读取成功 } } return 1; //返回读取失败 } ``` ```c #include "delay.h" static uint32_t TimingDelay; //延时初始化 void Delay_Init(void) { SysTick->CTRL = 0x00; SysTick->LOAD = SystemCoreClock / 1000000 - 1; SysTick->VAL = 0x00; SysTick->CTRL = 0x05; } //延时nms void Delay_Ms(uint16_t nms) { TimingDelay = nms; while(TimingDelay != 0); } //延时nus void Delay_Us(uint32_t nus) { uint32_t ticks; ticks = nus * (SystemCoreClock / 1000000); while(ticks--); } //SysTick中断处理函数 void SysTick_Handler(void) { if(TimingDelay != 0x00) { TimingDelay--; } } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值