jjzjj

STM32F1开发实例-振动传感器(机械)

苏墨实验室 2024-04-28 原文

振动(敲击)传感器

振动无处不在,有声音就有振动,哒哒的脚步是匆匆的过客,沙沙的夜雨是暗夜的忧伤。那你知道理科工程男是如何理解振动的吗?今天我们就来讲一讲本节的主角:最简单的机械式振动传感器。

下图即为振动传感器,外形有几种,原理都相同:1个弹簧电极+1个金属电极,振动时弹簧抖动碰到紧挨的金属电极,两个电极就会瞬间接通。我们通过单片机监测两电极的通断就可以知道当前是否发生振动,振动的频率是多少。原理就这么简单,准备上手!

 

一、硬件准备

  • 振动传感器1个,0.85元。
  • STM32F103核心板一块,某宝7块9。供电线跟你的手机充电线一样。
  • 面包板套件一套,某宝13.8元。不买套件,随手弄几根杜邦线也OK。
  • 仿真器,某宝7.5元。

   

 注:除了传感器以外,剩下的和其他实验共用,无需反复购买。

STM32F103核心板3.3V接面包板3.3V,GND接面包板GND;LED为板载,接口为PC13。

STM32F103核心板与振动传感器接线如下。

传感器引脚核心板引脚
S(信号输出)PA15
GNDGND
VCC3.3

二、软件准备

参见"STM32F1开发实例-开发环境搭建"。

三、实际效果

振动面包板,单片机侦测到传感器发出的信号,板载LED亮起1s后熄灭。

四、代码实现

项目工程源码如下:

https://download.csdn.net/download/wangpeng421/87548936

主程序代码如下:

//------引用头文件
#include "main.h"
#include "delay.h"
#include "timerPwm.h"
#include "led.h"
#include "ioSensor.h"

//---------------------------
//主函数
//---------------------------
int main(void) 
{  
	SysInit();									//系统初始化,如时钟等
	UserInit();  								//用户初始化,如传感器IO口等
	
	while(1)
	{			
		//-----高速代码执行区
	    ioSensorScan();						    //传感器扫描引脚信号
		if(ioSensor1.Status==_ON)				//如果传感器被触发		
		{
			__LED1_On(); 						//点亮led1
			led1.Status=_ON;                    //将led1状态设置为ON
		}
		//-----中速代码执行区.单位时间100us。
		if(SpeedCon_Num >= 10)//1ms进入1次
		{
			SpeedCon_Num=0;
			if(led1.Status==_ON)            //如果led1状态为ON
			{
				led1.onTime++;              //led1点亮计时
				if(led1.onTime>1000)		//1s后熄灭LED
				{
					led1.onTime=0;          //计时清零
					__LED1_Off(); 			//熄灭led1	
				  led1.Status=_Off;			//将led1状态设置为OFF
				}
			}
			else                            //led1没有亮起,则计时始终清零
				led1.onTime=0;
		}
	}
}

传感器驱动头文件:ioSensor.h

注:此类传感器均为简单的高低电平检测,所以编入IO触发类,通用ioSensor驱动。

//===============================================
//驱动名称:IO类传感器驱动
//公司名称: 
//代码编写: 苏墨 2020-05-12
//API 说明:1.头文件定义IO -> 初始化"io_SensorInit()" -> 调用"ioScan()"。
//			2.通过查询"ioSensor.Status"即可获取按键状态。按键状态值如下。
//			#define _OFF                 0x00          //释放
//			#define _ON                  0x01          //触发
//			#define _LongON              0x02          //长时间触发
//其他说明:
//===============================================
#ifndef __ioSensor_H__
#define __ioSensor_H__
#include "sys.h"	 

//-----IO定义
#define ioSensor1_PIN        GPIO_PIN_15					//传感器1
#define ioSensor1_Port       GPIOA
#define ioSensor2_PIN        GPIO_PIN_10					//传感器2
#define ioSensor2_Port       GPIOC
#define ioSensor3_PIN        GPIO_PIN_4						//传感器3
#define ioSensor3_Port       GPIOC
//------IO宏
#define __Enable_ioSensor_Port()				do{__HAL_RCC_GPIOA_CLK_ENABLE();__HAL_RCC_GPIOC_CLK_ENABLE();}while(0U)

//-----结构体
struct ioSensor{ 
	u8 onTime;					//触发计时.ON
	u8 offTime;					//释放计时.OFF
	u8 Status;					//状态
	u16 longOnTime;				//长触发计时.LongON
};

//-----常量
#define _OFF                     0x00          //释放
#define _ON                 	 0x01          //触发
#define _LongON            	 	 0x02          //长按


//-----变量
extern struct ioSensor ioSensor1;
extern struct ioSensor ioSensor2;
extern struct ioSensor ioSensor3;

//-----函数
void ioSensorInit();              //初始化
void ioSensorScan();              //扫描
#endif




传感器驱动代码:ioSensor.c

#include "ioSensor.h"

//-----常量


//-----变量
struct ioSensor ioSensor1;
struct ioSensor ioSensor2;
struct ioSensor ioSensor3;

//-----函数声明
void ioSensorInit();              //初始化
void ioSensorScan();              //扫描


//---------------------------
//初始化
//输入参数:无
//输出参数:无
//其他说明:
//---------------------------
void ioSensorInit()
{
  GPIO_InitTypeDef GPIO_Initure;

	__Enable_ioSensor_Port();
	GPIO_Initure.Pin=ioSensor1_PIN;
	GPIO_Initure.Mode=GPIO_MODE_INPUT;      		//输入模式
	GPIO_Initure.Pull=GPIO_PULLUP;                  //上拉
	GPIO_Initure.Speed=GPIO_SPEED_FREQ_HIGH;        //快速            
	HAL_GPIO_Init(ioSensor1_Port,&GPIO_Initure);
}

//---------------------------
//扫描
//输入参数:无
//输出参数:无
//其他说明:结构体将指明按键状态信息。详见.h文件。
//---------------------------
#define _OnTime         0         //触发计时.
#define _LongOnTime     1000      //长按计时.
#define _OffTime	    500       //释放计时
void ioSensorScan()
{
  //-----ioSensor1
  if(HAL_GPIO_ReadPin(ioSensor1_Port,ioSensor1_PIN)==1)
  {
		ioSensor1.offTime=0;
    ioSensor1.onTime++;
    if(ioSensor1.onTime>_OnTime)
    {
      ioSensor1.onTime=_OnTime;
      ioSensor1.Status=_ON; 
    }
  }
  else
  {
		//------------
		ioSensor1.offTime++;
		if(ioSensor1.offTime>_OffTime)
		{
			ioSensor1.onTime=0;
			ioSensor1.longOnTime=0;
			ioSensor1.Status=_OFF;
		}
  }
}

led驱动头文件:led.h

//===============================================
//驱动名称:led驱动
//公司名称: 
//代码编写: 苏墨 2020-05-12
//API 说明:1.头文件定义IO -> 初始化"ledInit()"。
//		    2.通过"led.Status"查询或设置led状态。状态值如下。
//			#define _OFF                 0x00          //释放
//			#define _ON                  0x01          //触发
//			通过“__LEDx_On()、__LEDx_Off()”控制对应LED亮灭。
//其他说明:
//===============================================
#ifndef __led_H__
#define __led_H__
#include "sys.h"	 


//-----IO定义
#define LED1_Pin        				GPIO_PIN_13
#define LED1_Port       				GPIOC
#define LED2_Pin        				GPIO_PIN_14
#define LED2_Port       				GPIOC
//-----宏
#define __Enable_LED_Port()			do{__HAL_RCC_GPIOC_CLK_ENABLE();}while(0U)
#define __LED1_On()     	  		do{HAL_GPIO_WritePin(LED1_Port,LED1_Pin,GPIO_PIN_RESET);}while(0U)
#define __LED1_Off()    				do{HAL_GPIO_WritePin(LED1_Port,LED1_Pin,GPIO_PIN_SET);}while(0U) 		
#define __LED2_On()     	  		do{HAL_GPIO_WritePin(LED1_Port,LED2_Pin,GPIO_PIN_RESET);}while(0U)		
#define __LED2_Off()    				do{HAL_GPIO_WritePin(LED1_Port,LED2_Pin,GPIO_PIN_SET);}while(0U) 

//-----结构体
struct led{ 
	u16 onTime;					//触发计时.ON
	u16 OffTime;				//释放计时.OFF
	u8 Status;					//状态
};

//-----常量



//-----变量
extern struct led led1;	


//-----函数
void ledInit();

#endif

led驱动代码:led.c

#ifndef __led_H__
#define __led_H__
#include "sys.h"	 


//-----IO定义
#define LED1_Pin        				GPIO_PIN_13
#define LED1_Port       				GPIOC
#define LED2_Pin        				GPIO_PIN_14
#define LED2_Port       				GPIOC
//-----宏
#define __Enable_LED_Port()			do{__HAL_RCC_GPIOC_CLK_ENABLE();}while(0U)
#define __LED1_On()     	  		do{HAL_GPIO_WritePin(LED1_Port,LED1_Pin,GPIO_PIN_RESET);}while(0U)
#define __LED1_Off()    				do{HAL_GPIO_WritePin(LED1_Port,LED1_Pin,GPIO_PIN_SET);}while(0U) 		
#define __LED2_On()     	  		do{HAL_GPIO_WritePin(LED1_Port,LED2_Pin,GPIO_PIN_RESET);}while(0U)		
#define __LED2_Off()    				do{HAL_GPIO_WritePin(LED1_Port,LED2_Pin,GPIO_PIN_SET);}while(0U) 

//-----结构体
struct led{ 
	u16 onTime;					//触发计时.ON
	u16 OffTime;				//释放计时.OFF
	u8 Status;					//状态
};

//-----常量



//-----变量
extern struct led led1;	


//-----函数
void ledInit();

#endif

有关STM32F1开发实例-振动传感器(机械)的更多相关文章

  1. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  2. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  3. ruby 正则表达式 - 如何替换字符串中匹配项的第 n 个实例 - 2

    在我的应用程序中,我需要能够找到所有数字子字符串,然后扫描每个子字符串,找到第一个匹配范围(例如5到15之间)的子字符串,并将该实例替换为另一个字符串“X”。我的测试字符串s="1foo100bar10gee1"我的初始模式是1个或多个数字的任何字符串,例如,re=Regexp.new(/\d+/)matches=s.scan(re)给出["1","100","10","1"]如果我想用“X”替换第N个匹配项,并且只替换第N个匹配项,我该怎么做?例如,如果我想替换第三个匹配项“10”(匹配项[2]),我不能只说s[matches[2]]="X"因为它做了两次替换“1fooX0barXg

  4. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  5. ruby-on-rails - Rails - 从另一个模型中创建一个模型的实例 - 2

    我有一个正在构建的应用程序,我需要一个模型来创建另一个模型的实例。我希望每辆车都有4个轮胎。汽车模型classCar轮胎模型classTire但是,在make_tires内部有一个错误,如果我为Tire尝试它,则没有用于创建或新建的activerecord方法。当我检查轮胎时,它没有这些方法。我该如何补救?错误是这样的:未定义的方法'create'forActiveRecord::AttributeMethods::Serialization::Tire::Module我测试了两个环境:测试和开发,它们都因相同的错误而失败。 最佳答案

  6. ruby-on-rails - RSpec:避免使用允许接收的任何实例 - 2

    我正在处理旧代码的一部分。beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)endRubocop错误如下:Avoidstubbingusing'allow_any_instance_of'我读到了RuboCop::RSpec:AnyInstance我试着像下面那样改变它。由此beforedoallow_any_instance_of(SportRateManager).toreceive(:create).and_return(true)end对此:let(:sport_

  7. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

  8. ruby - 在 Windows 机器上使用 Ruby 进行开发是否会适得其反? - 2

    这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby​​-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub

  9. ruby-on-rails - 在 Rails 开发环境中为 .ogv 文件设置 Mime 类型 - 2

    我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain

  10. ruby-on-rails - 使用 ruby​​ 将多个实例变量转换为散列的更好方法? - 2

    我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作:response.instance_variables.eachdo|r|my_hash.merge!(r.to_s.delete("@").intern=>response.instance_eval(r.to_s.delete("@")))end这有效,它将生成{:first="charlie",:last=>"kelly"},但它似乎有点hacky和不稳定。有更好的方法吗?编辑:我刚刚意识到我可以使用instance_variable_get作为该等式的第二部分,但这仍然是主要问题。

随机推荐