基于STM32F103C8T6(HAL库)的HC-SR501红外人体传感及HC-SR04超声波测距

系列文章目录

一、基于STM32F103C8T6最小系统板和STM32CubeMX实现LED灯循环闪烁
二、基于STM32F103C8T6和STM32CubeMX实现UART串口通信数据收发
三、实战小例程 基于STM32F103C8T6最小系统板和STM32CubeMX驱动WS2812B光立方
四、基于STM32F103C8T6最小系统板HAL库CubeMX驱动HC-SR501红外人体传感模块
五、基于STM32F103C8T6(HAL库)的HC-SR501红外人体传感及HC-SR04超声波测距



前言

我在上一篇文章中驱动了HC-SR501红外人体传感模块,但在测试过程中发现,这个模块的热释电探头过于灵敏了,甚至有的时候往上哈气,会因为检测到类似于人体的温度而误判为感应到人体,所以为了减少误判概率,我打算再添加一个测距模块,在红外感应到人体时,再次判断人体与传感器的距离,在满足预设的距离范围时,才确定有人体接近,进行后续操作。

一、模块简介

HC-SR501红外人体感应模块资料介绍:
探究人体红外传感器HC-SR501

HC-SR04超声波测距模块资料介绍:

在这里插入图片描述

引脚名称引脚作用
VCC3-5.5V供电(实测发现可以3.3V供电,但没有5V供电测得准)
GND接地
Trig外部触发信号输入,输入一个高于10μs的高电平即可触发模块测距
Echo回响信号输出,测距结束时此管脚输出一个高电平,电平宽度反映超声波往返时间之和

工作原理:

在这里插入图片描述
在这里插入图片描述

简而言之:
给Trig一个10μS以上的高电平,模块开始工作,模块内自动发送八个40khz方波,并自主检测是否有电波返回。此时需要检测Echo处的电平,当为高电平的时候记一个时间;当Echo出为低电平的时候再记一个时间,这两个时间的差就是高电平持续的时间,最后用测距公式进行计算

二、配置CubeMX

1、新建工程;
2、配置时钟源,在RCC里面的HSE配置的是晶振时钟;
3、配置程序烧录引脚SYS为SWD模式;
4、配置GPIO输出口,配置一个LED灯(我的板子是PC13),起到检测到人体时的指示作用;
5、配置GPIO输入口,用来读取HC-SR501模块的输出电平,我选的是PA1口;

6、配置GPIO输入口,用来检测HC-SR04超声波测距模块的回响信号输出,我选择PB5口,命名为HC_SR04_Echo_Pin;
7、配置GPIO输出口,用来触发HC-SR04超声波测距模块,我选择PB4口,命名为HC_SR04_Trig_Pin;
8、使能定时器,用来实现微秒延时,和测量HC-SR04模块高电平时间,我选择TIM2,并使能TIM2中断;

在这里插入图片描述
TIM2挂载在APB1,时钟来源频率为36MHz,故取预分频系数PSC=36-1,计数周期Counter设为60000-1。即此时定时器频率为36M/36=1MHz,每60ms进一次中断,60000μs对应实际测距范围大概为10m,远远超过我的需求。

在这里插入图片描述
在这里插入图片描述

9、配置串口收发引脚;
10、配置时钟树,我还是开到最高的72MHz;
11、进行项目设置,最后生成代码,CubeMX部分就大功告成了

在这里插入图片描述

三、硬件连线部分

CH340 ↔ STM32F103C8T6最小系统板:
 TX    ↔        RX
 RX    ↔        TX

DAP-LINK ↔ STM32F103C8T6最小系统板
  3.3V    ↔         VCC
  GND    ↔         GND
  SWIO   ↔         SWIO
 SWCLK  ↔        SWCLK

HC-SR501模块 ↔ STM32F103C8T6最小系统板
    OUT      ↔           PA1
    GND      ↔           GND

HC-SR501模块 ↔ DAP-LINK (这里注意模块供电范围为4.5V-20V,用3.3V无法驱动)
     VCC     ↔    5V

HC-SR04模块   ↔  CH340
     VCC      ↔    5V

HC-SR04模块   ↔ STM32F103C8T6最小系统板
     Trig       ↔          PB4
     Echo      ↔          PB5
     GND      ↔          GND

四、逻辑代码部分

代码框架是基于上一篇HC-SR501红外人体感应模块修改而成的,想要完整代码的话可以和上篇文章一起看。

HC_SR04.c

#include "main.h"
#include "HC_SR04.h"
#include "stm32f1xx_hal.h"
#include "stm32f1xx_it.h"
 
static float distance_result;
/*
*********************************************************
函数原型:void Delay_us(uint16_t time)
函数输入:无符号整形
函数输出:无
函数功能:利用定时器实现微秒级延时
*********************************************************
*/
void Delay_us(uint16_t time)
{
	uint16_t a1=TIM2->CNT;
	while(TIM2->CNT-a1<time);
}
/*
*********************************************************
函数原型:void HC_SR04_startrange(void)
函数输入:无
函数输出:无
函数功能:从trig引脚生成一个不小于10us的高电平触发测距,触发后模块自动产生8个40kHz方波,自动检测是否有信号返回
*********************************************************
*/
void HC_SR04_startrange(void)
{
	HAL_GPIO_WritePin(HC_SR04_Trig_GPIO_Port,HC_SR04_Trig_Pin,GPIO_PIN_SET);
	//HAL_Delay(5);
	Delay_us(10);
	HAL_GPIO_WritePin(HC_SR04_Trig_GPIO_Port,HC_SR04_Trig_Pin,GPIO_PIN_RESET);
	
}
/*
*********************************************************
函数原型:uint16_t HC_SR04_gettime(void)
函数输入:无
函数输出:无符号整型
函数功能:通过定时器获取当前时间
*********************************************************
*/
uint16_t HC_SR04_gettime(void)
{
	uint32_t a;
	a=TIM2->CNT;
	return a;
}
/*
*********************************************************
函数原型:float HC_SR04_getdistance(void)
函数输入:无
函数输出:浮点型
函数功能:获取与目标之间的距离
*********************************************************
*/
float HC_SR04_getdistance(void)
{   
	uint16_t time_node1;
 	uint16_t time_node2;
  	uint16_t measure;
	
  HC_SR04_startrange();
	
	TIM2->CNT = 0;
	
	//有信号返回则通过IO口Echo输出高电平,高电平持续时间即为超声波从发射到返回的时间,测试距离=( 高电平时间*声速(340m/s) )/ 2 
  	while(HAL_GPIO_ReadPin(HC_SR04_Echo_GPIO_Port,HC_SR04_Echo_Pin)==RESET);
	time_node1=HC_SR04_gettime();
	
	while(HAL_GPIO_ReadPin(HC_SR04_Echo_GPIO_Port,HC_SR04_Echo_Pin)==SET);
	time_node2=HC_SR04_gettime();
	
	measure=time_node2-time_node1;
	
	distance_result = measure * 17.0/1000;//距离=计数差值(us) / 1000000 * 340(m/s) * 100 / 2 = measure * 17/1000

  return distance_result;
}

HC_SR04.h

#ifndef _HC_SR04_H_
#define _HC_SR04_H_
void Delay_us(uint16_t time);
void HC_SR04_startrange(void);
float HC_SR04_getdistance(void);
uint16_t HC_SR04_gettime(void);
#endif

main.c

  /* USER CODE BEGIN 2 */
	HAL_TIM_Base_Start_IT(&htim2);
  /* USER CODE END 2 */
while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
		if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_1) == 1)
		{
			//HAL_Delay(50);
			HC_SR04_distance=HC_SR04_getdistance();
			HAL_GPIO_WritePin(GPIOC,GPIO_PIN_13,GPIO_PIN_RESET);//灭灯
			printf("有人,目前距离为:%.2f cm\r\n",HC_SR04_distance);
			HAL_Delay(100);
		}

		else  
		{
			HAL_GPIO_WritePin(GPIOC,GPIO_PIN_13,GPIO_PIN_SET);//亮灯
			printf("没人\r\n");
			HAL_Delay(1000);
		}
		
//		  printf("hellp\r\n");
//			HAL_Delay(1000);
  }
	
  /* USER CODE END 3 */
}

现象:

感应到人体,板载LED绿灯亮,同时串口打印测距结果。
在这里插入图片描述

注意:
重定义printf后,必须在target里面勾选上MicroLIB,调用一下这个微型库,不然一直卡在里面。

在这里插入图片描述

参考博客:
stm32f1驱动HC-SR04超声波测距模块

  • 17
    点赞
  • 218
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 6
    评论
以下是基于STM32F103C8T6HC-SR04实现超声波测距的代码,使用的是STM32CubeMX和Keil IDE进行开发: 1. 首先在STM32CubeMX中配置GPIO和定时器: GPIOA_Pin0:超声波传感器的Trig引脚 GPIOA_Pin1:超声波传感器的Echo引脚 TIM2_CH2:用于捕获超声波回波信号的定时器通道 2. 在Keil IDE中编写代码: #include "main.h" TIM_HandleTypeDef htim2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_TIM2_Init(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_TIM2_Init(); while (1) { uint32_t pulse_width = 0; uint32_t distance = 0; // 发送超声波信号 HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_SET); HAL_Delay(10); HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET); // 等待回波信号 while (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_1) == GPIO_PIN_RESET); uint32_t start_time = HAL_GetTick(); // 捕获回波信号 while (HAL_GPIO_ReadPin(GPIOA, GPIO_PIN_1) == GPIO_PIN_SET); uint32_t end_time = HAL_GetTick(); pulse_width = end_time - start_time; distance = pulse_width * 17 / 1000; // 声速340m/s,除以2得到往返时间,乘以17得到距离(单位:毫米) } } 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_HSE; RCC_OscInitStruct.HSEState = RCC_HSE_ON; RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9; 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_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK) { Error_Handler(); } } static void MX_TIM2_Init(void) { TIM_ClockConfigTypeDef sClockSourceConfig = {0}; TIM_MasterConfigTypeDef sMasterConfig = {0}; TIM_IC_InitTypeDef sConfigIC = {0}; htim2.Instance = TIM2; htim2.Init.Prescaler = 72 - 1; htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 0xFFFF; htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; if (HAL_TIM_Base_Init(&htim2) != HAL_OK) { Error_Handler(); } sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL; if (HAL_TIM_ConfigClockSource(&htim2, &sClockSourceConfig) != HAL_OK) { Error_Handler(); } if (HAL_TIM_IC_Init(&htim2) != HAL_OK) { Error_Handler(); } sConfigIC.ICPolarity = TIM_ICPOLARITY_RISING; sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI; sConfigIC.ICPrescaler = TIM_ICPSC_DIV1; sConfigIC.ICFilter = 0; if (HAL_TIM_IC_ConfigChannel(&htim2, &sConfigIC, TIM_CHANNEL_2) != HAL_OK) { Error_Handler(); } sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET; sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE; if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK) { Error_Handler(); } } static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; __HAL_RCC_GPIOA_CLK_ENABLE(); HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET); GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); GPIO_InitStruct.Pin = GPIO_PIN_1; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Pull = GPIO_PULLDOWN; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } void Error_Handler(void) { __disable_irq(); while (1) { } } 3. 在main函数中,首先发送超声波信号,等待回波信号,然后捕获回波信号,计算出距离。最后通过distance变量获取距离值。 注意事项: 1. 超声波传感器回波信号的宽度与距离成正比,需要使用定时器捕获回波信号的上升沿和下降沿,计算出回波信号的脉宽。 2. 超声波传感器的Trig引脚需要在发送超声波信号前置为高电平,发送完毕后置为低电平。 3. 超声波传感器的Echo引脚需要设置为输入模式,使用下拉电阻。 4. 定时器的时钟频率需要根据具体的系统时钟频率进行调整。 5. 超声波传感器的测量范围一般为2cm-4m,超出范围会出现误差。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

遗忘丶

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值