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

server/2025/3/1 23:58:20/

超声波技术的诞生灵感来源于大自然中的回声定位现象,尤其是蝙蝠的独特能力。蝙蝠通过发出高频超声波并捕捉回声来精确地探测周围的物体和猎物,即使在漆黑的夜晚也能轻松导航。
单片机中,也有着超声波这个模块,它在单片机上的标识是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();}
}

http://www.ppmy.cn/server/171657.html

相关文章

3D Web轻量化引擎HOOPS Communicator如何赋能航空航天制造?

在当今航空航天制造领域&#xff0c;精确度、效率和协作是推动行业发展的关键要素。随着数字化技术的飞速发展&#xff0c;3D Web可视化开发包HOOPS Communicator 为航空航天制造带来了革命性的变化。它凭借强大的功能和灵活的应用&#xff0c;助力企业在设计、生产、培训等各个…

【Web 大语言模型攻击简介】

Web 大语言模型攻击简介 一、攻击原理与分类二、检测与防御技术三、典型利用方式与案例四、防御建议与未来挑战总结 关于 Web 大语言模型攻击的原理、检测及利用方式的简介&#xff1a; 一、攻击原理与分类 提示注入&#xff08;Prompt Injection&#xff09; 核心机制&#xf…

AI算力革命重塑交换机市场:国产厂商的破局路径与战略高地

一、算力需求爆炸与网络架构重构 1.1 大模型训练对交换机的极限挑战 当前AI大模型参数量已突破十万亿级别&#xff0c;单次训练需协调数万张GPU卡。以GPT-4为例&#xff0c;其训练集群需要25,000块A100显卡&#xff0c;显存带宽达2TB/s&#xff0c;传统400G交换机每秒600GB的…

如何防止 Docker 注入了恶意脚本

根据您的描述&#xff0c;攻击者通过 CentOS 7 系统中的 Docker 注入了恶意脚本&#xff0c;导致自动启动名为 “masscan” 和 “x86botnigletjsw” 的进程。这些进程可能用于网络扫描或其他恶意活动。为了解决这一问题&#xff0c;建议您采取以下步骤&#xff1a; 1. 停止并删…

测试金蝶云的OpenAPI

如何使用Postman测试K3Cloud的OpenAPI 1. 引言 在本篇博客中&#xff0c;我将带你逐步了解如何使用Postman测试和使用K3Cloud的OpenAPI。内容包括下载所需的SDK文件、配置文件、API调用及测试等步骤。让我们开始吧&#xff01; 2. 下载所需的SDK文件 2.1 获取SDK 首先&…

蓝桥杯---快速排序(leetcode第159题)最小的k个元素(剑指offer原题)

文章目录 1.题目概述2.思路分析3.代码详解 1.题目概述 这个题目只是被包装了一下&#xff0c;本质上依然是使用的我们的快速排序算法&#xff0c;为什么这样说呢&#xff1f;因为仔细阅读题目你就会发现&#xff0c;这个需要我们去找到最小的前K个元素&#xff0c;并且进行返回…

Python毕业设计选题:基于django+vue的疫情数据可视化分析系统

开发语言&#xff1a;Python框架&#xff1a;djangoPython版本&#xff1a;python3.7.7数据库&#xff1a;mysql 5.7数据库工具&#xff1a;Navicat11开发软件&#xff1a;PyCharm 系统展示 管理员登录 管理员功能界面 用户管理 员工管理 疫情信息管理 检测预约管理 检测结果…

《SegFace: Face Segmentation of Long-Tail Classes》论文分享(侵删)

author{Kartik Narayan and Vibashan VS and Vishal M. Patel} 原文链接&#xff1a;[2412.08647] SegFace: Face Segmentation of Long-Tail Classes 摘要 人脸解析是指将人脸语义分割为眼睛、鼻子、头发等关键面部区域。它是各种高级应用程序的先决条件&#xff0c;包括人脸…