使用STM32 再实现感应开关盖垃圾桶

硬件介绍

SG90舵机


如上图所示的舵机SG90,橙线对应PWM信号,而PWM波的频率不能太高,大约50Hz,即周期0.02s,20ms左右。

在20ms的周期内,高电平占多少秒和舵机转到多少度的关系如下: 

0.5ms-----0度;2.5%对应函数中占空比为250

1.0ms-----45度;5.0%对应函数中占空比为500

1.5ms-----90度;7.5%对应函数中占空比为750

2.0ms-----135度;10.0%对应函数中占空比为1000

2.5ms-----180度;12.5%对应函数中占空比为1250

震动传感器

产生震动时,会输出低电平,绿色指示灯亮(开关信号指示灯)

蜂鸣器

 低电平触发

 超声波传感器

 使用方式见之前的博客

项目要求

检测靠近时,垃圾桶自动开盖并伴随滴一声,2秒后关盖

发生震动时,垃圾桶自动开盖并伴随滴一声,2秒后关盖

按下按键时,垃圾桶自动开盖并伴随滴一声,2秒后关盖

使用STM32控制舵机SG90

已知,舵机的驱动要使用周期20ms左右的PWM波,并通过调整占空比x来控制舵机的角度:

那如果我现在想要让舵机每隔1S转动一个角度,使得角度为0,45,90,135,180,0度。

那就可以使用我上节实现呼吸灯的CubeMX项目,对其进行修改:

1. 由于舵机中角度的计算是和高电平占周期的比例来换算的,所以要将CH Polarity改为High

2. 由于舵机需要的PWM的周期是20ms,则可以设置PSC = 7199, ARR = 199,这样当Tclk = 72M时,周期正好是0.02s,即20ms。

 3. 这就设置好了,更新项目并打开Keil,修改main.c中的main函数,CCRx为(ARR的199对应20ms,那0.5ms就对应199/40)约等于 5 时对应0度,10对应45度,15对应90度,20对应135度,25对应180度。

  HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_3); //打开Timer4的3号Channel

  while (1)
  {
	HAL_Delay(1000);
	__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度
		
	HAL_Delay(1000);
    __HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 10); //45度
		
	HAL_Delay(1000);
    __HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 15); //90度
		
	HAL_Delay(1000);
    __HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 20); //135度
		
	HAL_Delay(1000);
    __HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 25); //180度

  }

实现效果:(由于之前在用89C52做小车的时候,我已经把舵机和无线测距模块用热熔胶固定在小车上了,所以只看效果就可以) 

使用STM32控制无线测距模块HC-SR04

HC-SR04的开发逻辑在之前的章节里已经非常详细的介绍过了,这里直接开始实践:

Trig接入PB6Echo接入PB7,然后依然可以在刚刚舵机的CubeMX项目基础上修改:

1. 在之前的基础上,再使用一个TIM2,并且只用来作为计数功能,并软件控制何时停止计数,因此只需要设置TIM2的PSC而不需要设置ARR

值得一提的是,在89C52的使用中,驱动HC-SR04是Trig给至少10毫秒的高电平,所以我一开始是直接用HAL_Delay(20)来驱动的,但是我发现不行,所以很神奇的一件事情是,当使用STM32来驱动HC-SR04的时候,Trig的有效驱动又变回微秒级的了。。。

计数一次经过的时间是 (PSC + 1) / Tclk , 因此如果我想要计数1微秒,即0.000001s, 已知Tclk = 72 000 000, 那么PSC就应该设置为 71。然后在main.c中就可以定义出一个实现微秒级延时的函数:

//使用TIM2来做us级延时函数
void TIM2_Delay_us(uint16_t n_us)
{
/* 使能定时器2计数 */
__HAL_TIM_ENABLE(&htim2);
__HAL_TIM_SetCounter(&htim2, 0);
while(__HAL_TIM_GetCounter(&htim2) < ((1 * n_us)-1) );
/* 关闭定时器2计数 */
__HAL_TIM_DISABLE(&htim2);
}

然后,只要将PB6设置成GPIO_outputTrig是单片机发给HCSR04的信号), 将PB7设置成GPIO_inputEcho是HCSR04发回单片机的信号)就可以了:

2. 这就配置好了,HC-SR04的控制主要是在KEIL中自主实现的代码,其实思路和51的时候是一样一样的:

我现在想要实现的效果就是,当检测到距离小于5cm时,使得舵机转到135度,持续两秒然后回来:

void TIM2_Delay_us(uint16_t n_us)
{
/* 使能定时器2计数 */
    __HAL_TIM_ENABLE(&htim2);
    __HAL_TIM_SetCounter(&htim2, 0);
    while(__HAL_TIM_GetCounter(&htim2) < ((1 * n_us)-1) );
    /* 关闭定时器2计数 */
    __HAL_TIM_DISABLE(&htim2);
}

void StartHC()
{
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); //Trig写0
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET); //Trig写1
	TIM2_Delay_us(20); //持续20微妙
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); //Trig写0
	
}

void deal_dist()
{
	int cnt;
	float dist;
	
	StartHC();
	
	while((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7)) == GPIO_PIN_RESET); //等待Echo变高的一瞬间
	HAL_TIM_Base_Start(&htim2); //TIM2开始计时
	__HAL_TIM_SetCounter(&htim2,0); //将TIM2的计数器置0
	
	while((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7)) == GPIO_PIN_SET); //等待Echo变低的一瞬间	
	HAL_TIM_Base_Stop(&htim2); //TIM2停止计时
	
	cnt = __HAL_TIM_GetCounter(&htim2);//求出计了多少次,由于计数一次经过的时间是1us
	dist = cnt*340/2*0.000001*100;  //求出距离
	
	if(dist < 10){ //如果距离小于10cm
		__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 20); //135度
		HAL_Delay(2000);
		__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度
	}else{
		__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度
	}

}

int main(void)
{

  HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_3); //打开Timer4的3号Channel

  while (1)
  {
		deal_dist();
		HAL_Delay(200);
		
  }
}

实现效果:

感应开关盖垃圾桶实现

为了方便,就继续使用“mjm_test_PWM”的CubeMX项目文件!
在之前的基础上,要再额外加装震动传感器蜂鸣器,将震动传感器的D0接到PB5; 蜂鸣器的I/O接到PB4。

打开CubeMX修改:

1.添加GPIO口,并把PB4先拉高,并设置中断触发方式

 

 

 

 2. 打开中断,并设置优先级(把0的位置留给滴答定时器)

 

打开Keil修改:

打开stm32f1xx_it.c --> EXTI4(9_5)_IRQHandler() --> HAL_GPIO_EXTI_IRQHandler() --> HAL_GPIO_EXTI_Callback( )

HAL_GPIO_EXTI_Callback()就是中断处理程序,将他在main.c中重写:

注意,和电动车钥匙扣一样,因为要在中断函数中调用HAL_Delay,因此需要设置滴答定时器的优先级!!

void TIM2_Delay_us(uint16_t n_us)
{
	/* 使能定时器2计数 */
	__HAL_TIM_ENABLE(&htim2);
	__HAL_TIM_SetCounter(&htim2, 0);
	while(__HAL_TIM_GetCounter(&htim2) < ((1 * n_us)-1) );
	/* 关闭定时器2计数 */
	__HAL_TIM_DISABLE(&htim2);
}

void beep()
{
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_4, GPIO_PIN_RESET); //蜂鸣器响
	HAL_Delay(200);
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_4, GPIO_PIN_SET); //蜂鸣器停	
}

void StartHC()
{
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); //Trig写0
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET); //Trig写1
	//HAL_Delay(10);
	TIM2_Delay_us(20);
	HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); //Trig写0
	
}

void deal_dist()
{
	int cnt;
	float dist;
	
	StartHC();
	
	while((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7)) == GPIO_PIN_RESET); //等待Echo变高的一瞬间
	HAL_TIM_Base_Start(&htim2); //TIM2开始计时
	__HAL_TIM_SetCounter(&htim2,0); //将TIM2的计数器置0
	
	while((HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7)) == GPIO_PIN_SET); //等待Echo变低的一瞬间	
	HAL_TIM_Base_Stop(&htim2); //TIM2停止计时
	
	cnt = __HAL_TIM_GetCounter(&htim2);//求出计了多少次,由于计数一次经过的时间是1us
	dist = cnt*340/2*0.000001*100;  //求出距离
	
	if(dist < 10){ //如果距离小于10cm
		__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 20); //135度
		beep();
		HAL_Delay(2000);
		__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度
	}else{
		__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度
	}

}


void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
	if(GPIO_Pin == GPIO_PIN_5){ //震动导致的中断
		if(HAL_GPIO_ReadPin(GPIOB,GPIO_PIN_5) == GPIO_PIN_RESET){ //这个判断很重要
			__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 20); //135度
			beep();
			HAL_Delay(2000);
			//__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度 //不需要,因为main中的while一直再检测,如果震动之后,检测到有靠近,依然需要开盖;如果没有靠近,main里面的deal_dist函数也会关闭盖子
		}
	}
	
	if(GPIO_Pin == GPIO_PIN_0){ //按钮导致的中断
		HAL_Delay(50); //在检测到按键被按下的低电平的时候,先延迟50ms,再进行判断
		if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0) == GPIO_PIN_RESET){ //如果延迟过后依然是低电平
			__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 20); //135度
			beep();
			HAL_Delay(2000);
			//__HAL_TIM_SetCompare(&htim4, TIM_CHANNEL_3, 5); //0度 //不需要,因为main中的while一直再检测,如果按键之后,检测到有靠近,依然需要开盖;如果没有靠近,main里面的deal_dist函数也会关闭盖子
		}
	}
}


int main(void)
{
  HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_3); //打开Timer4的3号Channel
  HAL_NVIC_SetPriority(SysTick_IRQn,0,0); //必须写在" SystemClock_Config() "后面!!!

  while (1)
  {
		deal_dist();
		HAL_Delay(200); //这句延迟也很重要
  }

}

实现效果

可见,不管是距离小于10cm震动发生;还是按钮按下都可以触发舵机的转动,蜂鸣器的滴滴声,以及两秒后舵机的归位。(由于舵机在之前做小车的时候用热熔胶固定了,所以只要功能实现了就好,别太纠结长啥样!) 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/37032.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

msvcr110.dll丢失的修复教程,找不到msvcr110.dll解决办法哪个更推荐

msvcr110.dll是微软的Visual C运行库文件之一。它是Microsoft Visual Studio 2012的一部分&#xff0c;用于支持运行在Windows操作系统上使用Visual C编写的应用程序。在Windows系统中非常重要&#xff0c;如果丢失或是损坏就会造成很多程序无法启动运行。 会出现以下的报错提…

【云原生】k8s之存储卷

容器磁盘上的文件的生命周期是短暂的&#xff0c;这就使得在容器中运行重要应用时会出现一些问题。首先&#xff0c;当容器崩溃时&#xff0c;kubelet 会重启它&#xff0c;但是容器中的文件将丢失——容器以干净的状态&#xff08;镜像最初的状态&#xff09;重新启动。其次&a…

简单版本视频播放服务器V1

一直想做个家用版本的视频播放器&#xff0c;通过这个可以实现简单的电脑&#xff0c;通过浏览器就是可以访问电脑里面的视频&#xff0c;通过手机&#xff0c;平板等都是可以访问自己的视频服务了 后端代码&#xff1a; package mainimport ("fmt""io/iouti…

JMeter 中 3 种参数值的传递

目录 前言&#xff1a; (一) 从 CSV 文件读取要批量输入的变量 (二) 利用 Cookie 进行值的传递 (三) 利用正则匹配提取上一个接口的返回数据作为下个请求的输入 前言&#xff1a; 在JMeter中&#xff0c;参数值的传递是非常重要的&#xff0c;因为它允许你在测试过程中动态…

Proxy-Reflect使用详解

1 监听对象的操作 2 Proxy类基本使用 3 Proxy常见捕获器 4 Reflect介绍和作用 5 Reflect的基本使用 6 Reflect的receiver Proxy-监听对象属性的操作(ES5) 通过es5的defineProperty来给对象中的某个参数添加修改和获取时的响应式。 单独设置defineProperty是只能一次设置一…

Spring专家课程Day01_Spring-IOC

​ 文章目录 基础配置1)基础文件结构(Maven项目创建) 一、01_Spring概述_IOC_HelloWorld1.Spring框架的两个核心功能1.1) IOC/DI ,控制反转依赖注入!1.2) AOP,面向切面编程 2.IOC的两种模式2.1)配置文件中配置 Bean2.2)配置文件,组件扫描注解类注解Component 二、02_JavaBean_J…

Ajax简介和实例

目录 什么是 AJAX &#xff1f; AJAX实例 ajax-get无参 ajax-get有参 对象和查询字符串的互转 ajax-post ajax-post 表单 AJAX 是一种在无需重新加载整个网页的情况下&#xff0c;能够更新部分网页的技术。 什么是 AJAX &#xff1f; 菜鸟教程是这样介绍的&#xff1a…

本地Linux 部署 Dashy 并远程访问

文章目录 简介1. 安装Dashy2. 安装cpolar3.配置公网访问地址4. 固定域名访问 转载自cpolar极点云文章&#xff1a;本地Linux 部署 Dashy 并远程访问 简介 Dashy 是一个开源的自托管的导航页配置服务&#xff0c;具有易于使用的可视化编辑器、状态检查、小工具和主题等功能。你…

Gradle下载和配置教程:Windows、Mac和Linux系统安装指南

&#x1f337;&#x1f341; 博主 libin9iOak带您 Go to New World.✨&#x1f341; &#x1f984; 个人主页——libin9iOak的博客&#x1f390; &#x1f433; 《面试题大全》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33…

扒开 TCP 的外衣,看清 TCP 的本质

TCP 非常重要&#xff0c;它的内容很多&#xff0c;今天只能讲解其中的一部分&#xff0c;但足以让你超越 80 % 的编程开发人员对于 TCP 的认知。 本篇内容非常多&#xff0c;非常干&#xff0c;希望你花点时间仔细研究&#xff0c;我相信会对你有所帮助。 1. TCP 协议是什么…

云之道知识付费V2小程序V3.1.1独立平台版安装使用教程

据播播资源了解&#xff0c;云之道知识付费小程序是一款专注于知识付费的小程序源码&#xff0c;为内容创业者、自媒体和教育培训机构提供全方位的互联网解决方案。 由播播资源小编全套安装云之道知识付费V2独立版系统&#xff0c;系统支持无限多开&#xff0c;相比上几版出现…

Freertos-mini智能音箱项目---IO扩展芯片PCA9557

项目上用到的ESP32S3芯片引脚太少&#xff0c;选择了PCA9557扩展IO&#xff0c;通过一路i2c可以扩展出8个IO。这款芯片没有中断输入&#xff0c;所以更适合做扩展输出引脚用&#xff0c;内部寄存器也比较少&#xff0c;只有4个&#xff0c;使用起来很容易。 输入寄存器 输出寄存…

2021年美国大学生数学建模竞赛D题音乐的影响解题全过程文档及程序

2021年美国大学生数学建模竞赛 D题 音乐的影响 原题再现&#xff1a; 音乐是人类社会的一部分&#xff0c;是文化遗产的重要组成部分。 作为理解音乐在人类集体经验中所扮演角色的努力的一部分&#xff0c;我们被要求开发一种方法来量化音乐进化。 当艺术家创作一首新音乐时&…

运维小知识(二)——Linux大容量磁盘分区及挂载

centos系统安装&#xff1a;链接 目录 1.&#x1f353;&#x1f353;命令格式化磁盘 2.&#x1f353;&#x1f353;大容量硬盘分区 3.&#x1f353;&#x1f353;自动挂载 整理不易&#xff0c;欢迎一键三连&#xff01;&#xff01;&#xff01; 新系统装完之后&#xff0…

Go实现socks5服务器

SOCKS5 是一个代理协议&#xff0c;它在使用TCP/IP协议通讯的前端机器和服务器机器之间扮演一个中介角色&#xff0c;使得内部网中的前端机器变得能够访问Internet网中的服务器&#xff0c;或者使通讯更加安全。SOCKS5 服务器通过将前端发来的请求转发给真正的目标服务器&#…

【ubuntu重装系统后的软件配置_memo】

重装系统后系统环境恢复 备份安装系统常用的一些debvscode 更改sourcespip加速爬长城的家伙式儿安装ROS安装cmake安装git安装zsh顺便开个ssh提升幸福感的映射配置neovimplugins字体插件遇到的问题 锁键盘/鼠标小玩意儿 备份 实验时不起眼的图顺手写的脚本忘记从哪儿下载的资源…

#{} 和 ${} 的区别?

一、区别概述 1.1、主要区别&#xff1a; 1、#{} 是预编译处理&#xff0c;${} 是直接替换&#xff1b;2、${} 存在SQL注入的问题&#xff0c;而 #{} 不存在&#xff1b;Ps&#xff1a;这也是面试主要考察的部分~ 1.2、细节上&#xff1a; 1、${} 可以实现排序查询&#xff…

跨端技术栈综合考察:深入剖析 UniApp、Flutter、Taro 和 React Native 的优势与限制

文章目录 &#x1f4c8;UniApp⚡概念⚡优势⚡限制 &#x1f4c8;Flutter⚡概念⚡优势⚡限制 &#x1f4c8;Taro⚡概念⚡优势⚡限制 &#x1f4c8;React Native⚡概念⚡优势⚡限制 &#x1f4c8;跨端技术栈对比附录&#xff1a;「简历必备」前后端实战项目&#xff08;推荐&…

基于SpringBoot+Hadoop+Vue的企业网盘系统

完整资料进入【数字空间】查看——baidu搜索"writebug" 1.1.1 选题的背景 随着信息网络技术的迅猛发展&#xff0c;云计算技术从它的概念提出已经开始在实际生产环境中使用了。大部分的东西都已经慢慢云端化&#xff0c;这种新型的技术也受到许多互联网人员的关注&a…

【深入浅出 Spring Security(十二)】使用第三方(Github)授权登录

使用第三方&#xff08;Github&#xff09;授权登录 一、OAuth2 简单概述二、OAuth2 四种授权模式之授权码模式三、Github 授权登录准备工作创建 Spring Boot 项目Vue 测试代码测试效果 &#xff08;Github授权登录的具体操作在目录第三“章”&#xff09; 一、OAuth2 简单概述…