当前位置: 首页 > news >正文

探索超声波的奥秘——定时器与PCA

超声波技术的诞生灵感来源于大自然中的回声定位现象,尤其是蝙蝠的独特能力。蝙蝠通过发出高频超声波并捕捉回声来精确地探测周围的物体和猎物,即使在漆黑的夜晚也能轻松导航。
在单片机中,也有着超声波这个模块,它在单片机上的标识是JS1,在STC15F2K60S2单片机中,超声波位于如图所示的位置:
在这里插入图片描述
当成功编写超声波代码后,可以通过将手掌或者别的障碍物放在超声波水平线上达到测距的效果,接下来具体讲如何实现这个功能以及代码编写。

一、超声波距离计算公式

已知信息如下:超声波在空气中的传播速度为v(m/s),A发送超声波,经过时间t(s)传回A地,现在计算AB直接的直线距离s
时间t为A->B->A所消耗的时间,所以A->B所消耗的时间是 t 2 \frac{t}{2} 2t,根据距离公式易得s= v ∗ t 2 \frac{v*t}{2} 2vt

二、定义引脚

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
定义引脚P10为Tx,定义引脚P11为Rx。Rx和Tx的作用如下:

Rx作用Tx作用
0停止发送超声波0已接收到超声波
1开始发送超声波1未接收到超声波

使用超声波模块时,需要将跳线帽将板子上的1-3和2-4通过跳线帽短接,如下图所示:
在这里插入图片描述

三、实现步骤

不是
不是
通过Tx引脚发送8个40KHZ的方波
启动定时器计算计数脉冲
等待超声波返回
超声波信号是否返回
Rx=0
数据是否溢出
距离读取失败返回距离为0
停止计时并且计算距离

四、代码实现

1.发送8个40KHZ的方波

方波是一种周期性波形,它的特点是在每个周期内,波形的值要么处于高电平,要么处于低电平,并且在高低电平之间瞬时切换。
占空比:表示高电平持续的时间与一个周期总时间之比。通常,标准方波的占空比是 50%,即高电平和低电平各持续半个周期。
可以通过延迟函数来实现,T = 1 f \frac{1}{f} f1=25us,所以高电平/低电平所需的时间为12.5us,定义一个12us的延迟函数即可。

void Delay12us(void)	//@12.000MHz
{
	unsigned char data i;

	_nop_();
	_nop_();
	i = 33;
	while (--i);
}

void WaveInit(void)
{
	unsigned char i;
	for(i = 0; i < 8; i++)//连续发送8个40KHz的方波
	{
		Tx = 1;//开始发送超声波
		Delay12us();
		Tx = 0;//停止发送超声波
		Delay12us();
	}
}

2.定时器配置

本文给出两种定时器模式,一种是使用定时器0的TH0和TL1负责计时(一般来说,默认定时器0计数,定时器1计时),另一种是使用外挂定时器——PCA。
本文对PCA的原理不做介绍,只要读者知道如何配置定时器即可配置PCA。

使用PCA的优点:当一道题目考到超声波模块时,如果你使用定时器1去做的话,需要消耗一个定时器,如果又同时考到了NE555,你需要让定时器0去读取脉冲,所以你其他中断只能在定时器2中实现,而定时器2的配置和定时器0和1不同,需要消耗不必要的时间。

使用超声波只需要知道以下定时器内寄存器(位)的作用即可:

  • TMOD:定时器1配置为16位自动重装载
  • TL1、TH1:负责计时
  • TR1:TR1为高电平时开始计时
  • TF1:TF1为高电平时数据溢出
    其他寄存器查缺补漏,点击传送门:
    传送门:定时器专题——从理论到应用
    代码实现:
unsigned char Wave()
{
	unsigned int time;
	TMOD &= 0xF0;//配置定时器1为16位自动冲装载
	TL1 = TH1 = 0;//清空数据
	WaveInit();//发送方波信号
	TR1 = 1;//开始计时
	/*
	*当Rx为0(未接收到超声波)或者数据还未溢出时,继续等待
	*如果接收到了超声波,退出循环,进入后面的判断
	*如果还未接收到超声波但是数据已经溢出,也退出循环
	*只有当未接收到超声波而且数据未溢出时才继续等待
	*/
	while((Rx == 1) && (!TF1));
	TR1 = 0;//停止计时
	if(TF1)//数据溢出,本次读取超声波数据失败
	{
		TF1 = 0;//清除标志位
		return 0;
	}
	else //数据未溢出,读取超声波成功
	{
		time = (TH1 << 8) | TL0;
		return time*0.017;//v*t/2,由于时间以微秒为单位,1s=100 0000微秒,做一次进制转换得到0.017
	}
}

五、PCA配置

定时器配置中讲到使用到的寄存器(位)为TMOD,TH1,TL1,TR1,TF1,这些寄存器(位)在PCA中依次对应:COMD,CH,CL,CR,CF。
其中,CMOD寄存器功能如下图:
在这里插入图片描述
COMD的配置为0x00
所以copy一下上面写好的基于定时器1的超声波代码,可以得到以下基于PCA的超声波代码:

unsigned char Wave() //超声波距离读取函数
{
    unsigned int time;//时间储存变量
    CMOD =  0x00;//配置PCA工作模式
    CH = CL = 0;//复位计数值 等待超声波信号发出
    Wave_Init();//发送超声波信号
    CR = 1;//开始计时
    while((Rx == 1) && (CF == 0));//等待接受返回信号或者定时器溢出
    CR = 0;//停止计时
    if(CF == 0) //定时器没有溢出
    {
        time = CH << 8 | CL;//读取当前时间
        return (time * 0.017);//返回距离值
    }
    else
    {
        CF = 0;//清除溢出标志位
        return 0;
    }
}

六、应用

使用超声波将测量得到的数据显示在数码管的后三位上,超声波测距的结果以厘米为单位

Seg.h

#ifndef __Seg_H__
#define __Seg_H__

void SegDisp(unsigned char wela, unsigned char dula, unsigned char point);

#endif

Seg.c

#include <STC15F2K60S2.H>

code unsigned char Seg_Table[] =
{
0xc0, //0
0xf9, //1
0xa4, //2
0xb0, //3
0x99, //4
0x92, //5
0x82, //6
0xf8, //7
0x80, //8
0x90, //9
0xff, //空
0xdf, //-
0x8c, //P
0x8e //F
};

void SegDisp(unsigned char wela, unsigned char dula, unsigned char point)
{
	P0 = 0xff;
	P2 = P2 & 0x1f | 0xe0;
	P2 &= 0x1f;
	
	P0 = (0x01 << wela);
	P2 = P2 & 0x1f | 0xc0;
	P2 &= 0x1f;
	
	P0 = Seg_Table[dula];
	P2 = P2 & 0x1f | 0xe0;
	P2 &= 0x1f;
}

main.c

#include <STC15F2K60S2.H>
#include "Init.h"
#include "LED.h"
#include "Key.h"
#include "Seg.h"

/* 变量声明区 */
unsigned char Key_Slow; //按键减速变量 10ms 
unsigned char Key_Val, Key_Down, Key_Up, Key_Old; //按键检测四件套
unsigned int Seg_Slow; //数码管减速变量 500ms
unsigned char Seg_Buf[] = {10,10,10,10,10,10,10,10,10,10};//数码管缓存数组
unsigned char Seg_Pos;//数码管缓存数组专用索引
unsigned char Seg_Point[8] = {0,0,0,0,0,0,0,0};//数码管小数点使能数组
unsigned char ucLed[8] = {0,0,0,0,0,0,0,0};//LED显示数据存放数组
unsigned char wave;
/* 按键处理函数 */
void Key_Proc()
{
	if(Key_Slow) return;
	Key_Slow = 1; //按键减速
	
	Key_Val = Key();
	Key_Down = Key_Val & ~Key_Old;	 
	Key_Up = ~Key_Val & Key_Old;
	Key_Old = Key_Val;
	
}

/* 信息处理函数 */
void Seg_Proc()
{
	if(Seg_Slow) return;
	Seg_Slow = 1; //数码管减速

	
	SegBuf[0] = 10;
	SegBuf[1] = 10;
	SegBuf[2] = 10;
	SegBuf[3] = 10;
	SegBuf[4] = 10;
	SegBuf[5] = wave / 100;
	SegBuf[6] = wave / 10 % 10;
	SegBuf[7] = wave % 10;				
}

/* 其他显示函数 */
void Led_Proc()
{

}

/* 定时器1用于计时 */
void Timer1_Init(void)		//1毫秒@12.000MHz
{
	AUXR &= 0xBF;			//定时器时钟12T模式
	TMOD &= 0x0F;			//设置定时器模式
	TL1 = 0x18;				//设置定时初始值
	TH1 = 0xFC;				//设置定时初始值
	TF1 = 0;				//清除TF1标志
	TR1 = 1;				//定时器1开始计时
    ET1 = 1;
	EA = 1;
}

/* 定时器1中断服务函数 */
void Timer1_Server() interrupt 3
{
	if(++Key_Slow == 10) Key_Slow = 0; //按键延迟
	if(++Seg_Slow == 100) Seg_Slow = 0; //数码管延迟
	if(++Seg_Pos == 8) Seg_Pos = 0;	   //数码管显示
}

void main()
{
	Init();
	Timer0_Init();
	Timer1_Init();
	while(1)
	{
		Key_Proc(); 
		Seg_Proc();
		Led_Proc();
	}
}

相关文章:

  • 阳光高考瑞数6vmp算法还原
  • DeepSeek赋能大模型内容安全,网易易盾AIGC内容风控解决方案三大升级
  • Axios 取消请求
  • 微软推出Office免费版,限制诸多,只能编辑不能保存到本地
  • Ubuntu搭建esp32环境 配置打开AT指令集 websocket功能
  • C++引用
  • 【Deepseek+Browser-Use搭建 Web UI自动化】
  • AWS SDK for Java 1.x 403问题解决方法和原因
  • 【学习笔记】Kubernetes
  • React加TypeScript最新部署完整版
  • 系统定时器SysTick
  • Spring 源码硬核解析系列专题(七):Spring Boot 与 Spring Cloud 的微服务源码解析
  • 【前端进阶】07 http协议和前端开发有什么关系
  • springboot实现文件上传到华为云的obs
  • Android 12.0 第三方app接收不到开机广播问题的解决以及开机自启动功能实现一
  • Java+SpringBoot+Vue+数据可视化的音乐推荐与可视化平台(程序+论文+讲解+安装+调试+售后)
  • 为什么一个ip地址可以用浏览器打开,但是不能ping通
  • 算法题训练 ——— NC313 两个数组的交集
  • vscode中使用PlatformIO创建工程加载慢
  • 优选算法大集合(待更新)
  • 咖搭姆少儿编程加盟/seo关键词怎么优化
  • java web是做网站的吗/域名买卖交易平台
  • wordpress主页不显示/吉安seo网站快速排名
  • 微云影视自助建站系统/河南网站seo靠谱
  • 赣州市微程网络科技有限公司/北京网站营销seo方案
  • 哪个平台建网站比较好/百度有刷排名软件