基于c51单片机超声波测距仪

2023-10-09 14:40

本文主要是介绍基于c51单片机超声波测距仪,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        整个系统由AT89C51,超声波电路,显示电路和报警电路,按键控制组成,系统复位后,首先对各模块进行初始化,初始化后根据超声波模块返回的回波进行数据计算,把数据显示到LMO16L液晶显示器上,并与设定的报警值相比较,小于报警值则蜂鸣器响起、指示灯亮,延时判断是否有按键输入修改报警值,修改报警值后继续进行读取超声波测距。

##超声波模块

       超声波是由机械振动产生的,可在不同介质中以不同的速度传播,具有良好的定向性,能量集中,传输过程衰减较小,发射能力强等优点。超声波测距是通过不断检测超声波发射后遇到障碍物所反射回来的回波,从而测出从发射到接受回波的时间差T,在速度V已知的情况下,距离S = V*T/2.

 

1.HC-SR04模块一共有四个引脚,分别是VCC 、TRIG(触发器引脚)、ECHO(接受引脚)、GUD(地线)。

2.基本工作原理:采用IO口TRIG触发测距,至少给10us的高电平信号;模块自动发送8个40khz的方波,自动检测是否有信号返回;有信号返回ECHO输出一个高电平,电平持续的时间就是超声波发射到返回的时间。测试距离=(高电平时间*声速(340m/s))/2

 3.定时器计算时间

将TH0和TL0拼接为16进制,需将TH0左移8位

二进制左移一位,01变01(2)   相当于乘2 

二进制左移8位,相当于乘以2的8次方=256

time = (TH0*256 + TL0);.//单位为us

##LM016L液晶显示 

 

 ##protues仿真图

 ##主流程图

#include <reg52.h>
#include <intrins.h>#define uchar unsigned char	// 以后unsigned char就可以用uchar代替
#define uint  unsigned int	// 以后unsigned int 就可以用uint 代替sfr ISP_DATA  = 0xe2;			// 数据寄存器
sfr ISP_ADDRH = 0xe3;			// 地址寄存器高八位
sfr ISP_ADDRL = 0xe4;			// 地址寄存器低八位
sfr ISP_CMD   = 0xe5;			// 命令寄存器
sfr ISP_TRIG  = 0xe6;			// 命令触发寄存器
sfr ISP_CONTR = 0xe7;			// 命令寄存器sbit LcdRs_P   = P1^1;		// 1602液晶的RS管脚       
sbit LcdRw_P   = P1^2;		// 1602液晶的RW管脚 
sbit LcdEn_P   = P1^3;		// 1602液晶的EN管脚sbit Trig4_P   = P1^4;		// 超声波模块4的Trig管脚
sbit Echo4_P   = P1^5;		// 超声波模块4的Echo管脚
sbit KeySet_P  = P2^2;		// 设置按键的管脚
sbit KeyDown_P = P2^1;		// 减按键的管脚
sbit KeyUp_P   = P2^0;		// 加按键的管脚
sbit Buzzer_P  = P2^3;		// 蜂鸣器的管脚sbit Led4_P    = P1^0;		// 传感器4报警灯uint gAlarm;							// 报警距离变量/*********************************************************/
// 单片机内部EEPROM不使能
/*********************************************************/
void ISP_Disable()
{ISP_CONTR = 0;ISP_ADDRH = 0;ISP_ADDRL = 0;
}/*********************************************************/
// 从单片机内部EEPROM读一个字节,从0x2000地址开始
/*********************************************************/
unsigned char EEPROM_Read(unsigned int add)
{ISP_DATA  = 0x00;ISP_CONTR = 0x83;ISP_CMD   = 0x01;ISP_ADDRH = (unsigned char)(add>>8);ISP_ADDRL = (unsigned char)(add&0xff);// 对STC89C51系列来说,每次要写入0x46,再写入0xB9,ISP/IAP才会生效ISP_TRIG  = 0x46;	   ISP_TRIG  = 0xB9;_nop_();ISP_Disable();return (ISP_DATA);
}/*********************************************************/
// 往单片机内部EEPROM写一个字节,从0x2000地址开始
/*********************************************************/
void EEPROM_Write(unsigned int add,unsigned char ch)
{ISP_CONTR = 0x83;ISP_CMD   = 0x02;ISP_ADDRH = (unsigned char)(add>>8);ISP_ADDRL = (unsigned char)(add&0xff);ISP_DATA  = ch;ISP_TRIG  = 0x46;ISP_TRIG  = 0xB9;_nop_();ISP_Disable();
}/*********************************************************/
// 擦除单片机内部EEPROM的一个扇区
// 写8个扇区中随便一个的地址,便擦除该扇区,写入前要先擦除
/*********************************************************/
void Sector_Erase(unsigned int add)	  
{ISP_CONTR = 0x83;ISP_CMD   = 0x03;ISP_ADDRH = (unsigned char)(add>>8);ISP_ADDRL = (unsigned char)(add&0xff);ISP_TRIG  = 0x46;ISP_TRIG  = 0xB9;_nop_();ISP_Disable();
}/*********************************************************/
// 毫秒级的延时函数,time是要延时的毫秒数
/*********************************************************/
void DelayMs(uint time)
{uint i,j;for(i=0;i<time;i++)for(j=0;j<112;j++);
}/*********************************************************/
// 1602液晶写命令函数,cmd就是要写入的命令
/*********************************************************/
void LcdWriteCmd(uchar cmd)
{ LcdRs_P = 0;LcdRw_P = 0;LcdEn_P = 0;P0=cmd;DelayMs(2);LcdEn_P = 1;    DelayMs(2);LcdEn_P = 0;	
}/*********************************************************/
// 1602液晶写数据函数,dat就是要写入的数据
/*********************************************************/
void LcdWriteData(uchar dat)
{LcdRs_P = 1; LcdRw_P = 0;LcdEn_P = 0;P0=dat;DelayMs(2);LcdEn_P = 1;    DelayMs(2);LcdEn_P = 0;
}/*********************************************************/
// 液晶光标定位函数
/*********************************************************/
void LcdGotoXY(uchar line,uchar column)
{// 第一行if(line==0)        LcdWriteCmd(0x80+column); // 第二行if(line==1)        LcdWriteCmd(0x80+0x40+column); 
}/*********************************************************/
// 液晶输出字符串函数
/*********************************************************/
void LcdPrintStr(uchar *str)
{while(*str!='\0')LcdWriteData(*str++);
}/*********************************************************/
// 液晶输出数字
/*********************************************************/
void LcdPrintNum(uint num)
{LcdWriteData(num/100+0x30);				// 百位LcdWriteData(num%100/10+0x30);		// 十位LcdWriteData(num%10+0x30);				// 个位
}/*********************************************************/
// 1602液晶功能初始化
/*********************************************************/
void LcdInit()
{LcdWriteCmd(0x38);        // 16*2显示,5*7点阵,8位数据口LcdWriteCmd(0x0C);        // 开显示,不显示光标LcdWriteCmd(0x06);        // 地址加1,当写入数据后光标右移LcdWriteCmd(0x01);        // 清屏
}/*********************************************************/
// 1602液晶显示内容初始化
/*********************************************************/
void LcdShowInit()
{LcdGotoXY(0,0);	    				// 定位到第0行第0列LcdPrintStr("L:       cm");	// 第0行显示“      U         ”LcdGotoXY(1,0);	    							// 定位到第1行第0列}/*********************************************************/
// 计算传感器4测量到的距离
/*********************************************************/
uint GetDistance4(void)
{uint ss;					// 用于记录测得的距离TH0=0;TL0=0;Trig4_P=1;				// 给超声波模块4一个开始脉冲DelayMs(1);Trig4_P=0;while(!Echo4_P);	// 等待超声波模块4的返回脉冲TR0=1;						// 启动定时器,开始计时while(Echo4_P);		// 等待超声波模块4的返回脉冲结束TR0=0;						// 停止定时器,停止计时ss=((TH0*256+TL0)*0.034)/2+1;		// 距离cm=(时间us * 速度cm/us)/2return ss;
}/*********************************************************/
// 按键扫描
/*********************************************************/
void KeyScanf()
{if(KeySet_P==0)					// 判断是否有按键按下{LcdGotoXY(0,0);	    							// 光标定位LcdPrintStr("   Alarm Set    ");	// 第0行显示“   Alarm Set    ”LcdGotoXY(1,0);										// 光标定位LcdPrintStr("  alarm=   cm   ");	// 第1行显示“  alarm=   cm   ”LcdGotoXY(1,8);										// 光标定位LcdPrintNum(gAlarm);							// 显示当前的报警值DelayMs(10);											// 消除按键按下的抖动while(!KeySet_P);									// 等待按键释放DelayMs(10);											// 消除按键松开的抖动		while(1){		/* 报警值减的处理 */if(KeyDown_P==0)					{if(gAlarm>2)						// 报警值大于2才能减1gAlarm--;							// 报警值减1LcdGotoXY(1,8);					// 光标定位LcdPrintNum(gAlarm);		// 刷新修改后的报警值DelayMs(300);						// 延时}/* 报警值加的处理 */if(KeyUp_P==0)					{if(gAlarm<400)					// 报警值小于400才能加1gAlarm++;							// 报警值加1LcdGotoXY(1,8);					// 光标定位LcdPrintNum(gAlarm);		// 刷新修改后的报警值DelayMs(300);						// 延时}/* 退出报警值设置 */if(KeySet_P==0)					{break;									// 退出while循环}	}LcdShowInit();							// 液晶恢复测量到测量界面DelayMs(10);		  					// 消除按键按下的抖动while(!KeySet_P);						// 等待按键释放DelayMs(10);		  					// 消除按键松开的抖动		Sector_Erase(0x2000);				// 保存报警距离EEPROM_Write(0x2000,gAlarm/100);EEPROM_Write(0x2001,gAlarm%100);		}	
}/*********************************************************/
// 传感器4报警判断
/*********************************************************/
void AlarmJudge4(uint ss)
{if(ss<gAlarm)		// LED灯判断{Led4_P=0;}else{Led4_P=1;}if(Led4_P==0)	// 蜂鸣器判断{Buzzer_P=0;}else{Buzzer_P=1;}
}/*********************************************************/
// 报警值初始化
/*********************************************************/
void AlarmInit()
{gAlarm=EEPROM_Read(0x2000)*100+EEPROM_Read(0x2001);		// 从EEPROM读取报警值if((gAlarm==0)||(gAlarm>400))			// 如果读取到的报警值异常(等于0或大于400则认为异常){gAlarm=25;											// 重新赋值报警值为25}
}/*********************************************************/
// 主函数
/*********************************************************/
void main()
{uchar i;									// 循环变量uint dist;								// 保存测量结果LcdInit();								// 液晶功能初始化LcdShowInit();						// 液晶显示内容初始化AlarmInit();							// 报警值初始化TMOD = 0x01;							// 选择定时器0,并且确定是工作方式1(为了超声波模块测量距离计时用的)Trig4_P=0;while(1){		/*传感器4*/dist=GetDistance4();		// 读取超声波模块4测量到的距离LcdGotoXY(0,2);	    		// 光标定位LcdPrintNum(dist);			// 显示传感器4测量到的距离AlarmJudge4(dist);			// 判断传感器4的测量距离是否需要报警/*延时并扫描按键*/for(i=0;i<15;i++){KeyScanf();DelayMs(10);}}
}

这篇关于基于c51单片机超声波测距仪的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/173538

相关文章

基于51单片机的自动转向修复系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机

单片机毕业设计基于单片机的智能门禁系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍程序代码部分参考 设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订

基于stm32的河流检测系统-单片机毕业设计

文章目录 前言资料获取设计介绍功能介绍具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机设计精品

VB和51单片机串口通信讲解(只针对VB部分)

标记:该篇文章全部搬自如下网址:http://www.crystalradio.cn/thread-321839-1-1.html,谢谢啦            里面关于中文接收的部分,大家可以好好学习下,题主也在研究中................... Commport;设置或返回串口号。 SettingS:以字符串的形式设置或返回串口通信参数。 Portopen:设置或返回串口

超声波清洗机哪个品牌比较好一点的?清洁力强的超声波清洗机品牌

随着生活水平的不断提升和幸福感的增强,珠宝、饰品和眼镜等物品已成为许多家庭的常备之物。然而,这些贵重细小的物件易于积聚微尘与隐形细菌,长此以往可能悄悄影响家人的健康,毕竟细菌是肉眼难以察觉的隐患。超声波清洗机应运而生,它以高科技手段有效地解决了这一隐忧,深层清洁,守护家人免受微小污染物的潜在威胁。不过现在市面上超声波清洗机品牌挺多的,究竟有哪些品牌的超声波清洗机比较好一点呢?接下来就为大家带来四款

MCU5.51单片机的最小系统

1.最小系统的组成部分 晶振电路(时钟),复位电路,电源电路(控制电压,保持稳定),下载电路(外加的,用于烧录程序) 烧录: 通过下载电路,把程序下载到单片机中用于运行 2.晶振电路 电路图: 是晶振 是电容 为了防止晶振起振时产生影响,因此加电容 3.复位电路 电路图: 按按钮RSTK1(保持一段时间的高电平)即可实现STC芯片的复位 4.电源电路 电路图:

单片机XTAL引脚引出的晶振分析

51单片机的18,19脚XTAL1,XTAL2用来提供外部振荡源给片内的时钟电路。 XTAL1和XTAL2引脚,该单片机可以使用外部时钟也可以使用内部时钟。 当使用内部时钟时,此二引线端用于外接石英晶体和微调电容; 当使用外部时钟时,用于接外部时钟信号,NMOS接XTAL2,CMOS接XTAL1。 原理: XTAL1和XTAL2分别是一个反相器的输入和输出。NMOS的反相器是

树莓派使用WiringPi库配合时间函数实现超声波测距

树莓派使用WiringPi库配合时间函数实现超声波测距 文章目录 树莓派使用WiringPi库配合时间函数实现超声波测距一、HR-04超声波模块原理1.1 超声波测距原理:1.2 超声波时序图: 二、树莓派与超声波模块硬件连接三、时间函数3.1 时间函数gettimeofday()原型和头文件: 四、实现超声波测距4.1 使用wiringOP库和时间函数实现超声波测距: 五、实现超声波测距

飞凌单片机开发IDE环境

飞凌单片机 资料下载选择芯片类型:MCU->GPIO选择芯片型号:FM8PE53B IDE工具安装: 资料下载 飞凌官网 选择芯片类型:MCU->GPIO 选择芯片型号:FM8PE53B IDE工具安装: 运行Generic ICE 0.02-250.03 setup,一直点击“下一步”即可。 双击运行:Generic ICE

基于51单片机的智能小车转向控制系统设计与实现

文章目录 前言资料获取设计介绍功能介绍具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订阅👇🏻 单片机设计精品