51单片机超声波测距程序详解
21ic 2025-06-25

51单片机超声波测距程序详解

超声波四通道测距:超声波测距实现分为三大块:

其一是12864带字库的液晶驱动程序:


代码如下:

/////////////////12864驱动程序///////////////////////////

//1写数据

void WriteDataLCD(unsigned char WDLCD)

{

ReadStatusLCD(); //检测忙

LCD_RS = 1;

LCD_RW = 0;

LCD_Data = WDLCD;

LCD_E = 1;

LCD_E = 1;

LCD_E = 1;

LCD_E = 0;

}

//2写指令

void WriteCommandLCD(unsigned char WCLCD,BuysC) //BuysC为0时忽略忙检测

{

if (BuysC) ReadStatusLCD(); //根据需要检测忙

LCD_RS = 0;

LCD_RW = 0;

LCD_Data = WCLCD;

LCD_E = 1;

LCD_E = 1;

LCD_E = 1;

LCD_E = 0;

}

//3读数据

unsigned char ReadDataLCD(void)

{

LCD_RS = 1;

LCD_RW = 1;

LCD_E = 0;

LCD_E = 0;

LCD_E = 1;

return(LCD_Data);

}

//4读状态

unsigned char ReadStatusLCD(void)

{

LCD_Data = 0xFF;

LCD_RS = 0;

LCD_RW = 1;

LCD_E = 1;

while (LCD_Data & Busy); //检测忙信号

LCD_E = 0;

return(LCD_Data);

}

void LCDInit(void) //5LCM初始化

{

WriteCommandLCD(0x30,1); //显示模式设置,开始要求每次检测忙信号

WriteCommandLCD(0x01,1); //显示清屏

WriteCommandLCD(0x06,1); // 显示光标移动设置

WriteCommandLCD(0x0C,1); // 显示开及光标设置

}

void LCDClear(void) //6清屏

{

WriteCommandLCD(0x01,1); //显示清屏

WriteCommandLCD(0x34,1); // 显示光标移动设置

WriteCommandLCD(0x30,1); // 显示开及光标设置

}

void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)

{

if(Y<1)

Y=1;

if(Y>4)

Y=4;

X &= 0x0F; //限制X不能大于16,Y不能大于1

switch(Y){

case 1:X|=0X80;break;

case 2:X|=0X90;break;

case 3:X|=0X88;break;

case 4:X|=0X98;break;

}

WriteCommandLCD(X, 0); //这里不检测忙信号,发送地址码

WriteDataLCD(DData);

}

//9按指定位置显示一串字符

void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)

{

unsigned char ListLength,X2;

ListLength = 0;

X2=X;

if(Y<1)

Y=1;

if(Y>4)

Y=4;

X &= 0x0F; //限制X不能大于16,Y在1-4之内

switch(Y){

case 1:X2|=0X80;break; //根据行数来选择相应地址

case 2:X2|=0X90;break;

case 3:X2|=0X88;break;

case 4:X2|=0X98;break;

}

WriteCommandLCD(X2, 1); //10发送地址码

while (DData[ListLength]>=0x20) //若到达字串尾则退出

{

if (X <= 0x0F) //X坐标应小于0xF

{

WriteDataLCD(DData[ListLength]); //

ListLength++;

X++;

Delay5Ms();

}

}

}

其二是超声波模块的检测部分程序:该部分程序的超声波发射与接受程序没有采用外部中断的方式实现毕竟51单片机的外部中断稀少比较珍贵,也最多只能驱动2组,达不到预期所需要的。故在本程序中我是才用的端口扫描的方式实现的。

程序代码如下:void tran4(void)///可通过设置K数值选择执行相应的I0发射超声波

{

uchar i;

float temp;

TMOD=0XF0;

TMOD |=0X01;

TL0=0X47;

TH0=0XFF;

TR0=0;

ET0=0;//关闭timme0中断?

csb4=1;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

csb4=0;

TR0=0;

TH0=0;

TL0=0;

while(in4==0);

TR0=1;

while(in4==1);

TR0=0;

temp=(TH0*256+TL0)*1.08/58; //1.08为分频之后的数

//蜂鸣器响

if(temp<20)beef=0;

delay_nms(60);

beef=1;

//

if(temp>4000)temp=4000;

dis =(unsigned int)temp;//dis将直接用于显示函数

delay_nms(60);

display(dis);

shuju[0]=shuju[0]+3;

for(i=0;i<6;i++)

DisplayOneChar(i,4,shuju[i]); //显示字库中的中文数子

}

超声波测距需要:现发送一组超声波,当超声波受到障碍物被弹回后经超声波接受线路接受后根据高低电平进行计算这段时间,并将该时间与超声波传输的速度之间的关系换算成与物体之间的距离:

这就是其三:将接受到发挥的超声波所需时间转换程与物体之间的距离:

换算程序如下:

void display(uint dat)

{

uchar i,j,k;//定义变量

i=dat/100;//白

j=dat0/10;//十位

k=dat0;//个

shuju[0]=65;

shuju[1]=i+48;//根据asc码值要加48才能正确在液晶上显示字符(具体参考液晶手册)

shuju[2]='.';

shuju[3]=j+48;

shuju[4]=k+48;

shuju[5]=109;//或者shuju[4]='m';

}

以上三步完成程序也就相当容易了:

完整超声波四通道端口扫描方式测距程序如下:

#include

#include

#include

#define uchar unsigned char

#define uint unsigned int

#define nop _nop_()

#define LCD_Data P0

#define Busy 0x80 //用于检测LCD状态字中的Busy标识

sbit csb1=P1^0;//tr 40KHz超声波发射脚

sbit in1=P1^1;

sbit csb2=P1^2;

sbit in2=P1^3;

sbit csb3=P2^2;

sbit in3=P2^3;

sbit in4=P2^0;

sbit csb4=P2^1;

sbit LCD_RS=P3^5;//定义引脚

sbit LCD_RW=P3^6;

sbit LCD_E=P3^4;

//sbit PSB =P3^7;

sbit PSB =P3^1; //PSB脚为12864-12系列的串、并通讯功能切换,我们使用8位并行接口,PSB=1

sbit beef =P2^6;

//LCD显示字符函数

void WriteDataLCD(unsigned char WDLCD);

void WriteCommandLCD(unsigned char WCLCD,BuysC);

unsigned char ReadDataLCD(void);

unsigned char ReadStatusLCD(void);

void LCDInit(void);

void LCDClear(void);

//void LCDFlash(void);

void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData);

void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData);

//void DisplayImage (unsigned char code *DData);

void Delay5Ms(void);

void Delay400Ms(void);

/////////////////////////////超声波发射接收子函数体

void delay100us();

void delay_nms(uint n);

void tran1(void);

void tran2(void);

void tran3(void);

void tran4(void);

void display(uint dat);

unsigned char shuju[5];//////////////////////////////字符定义;

////////////////////////////////////////////////超声波发射接收全局变量

unsigned char flag;

unsigned int tmp,dis,high_time ,low_time ;

///////////////////////////////////////////////

/////////////////////////////////////////////////主程序函数

void main(void)

{

Delay400Ms(); //启动等待,等LCD讲入工作状态

LCDInit(); //LCM初始化

Delay5Ms(); //延时片刻(可不要)

while(1)

{

// uint i;

tran1( );

// Delay400Ms();

// Delay400Ms();//通过延时防止因csb1接收引起中断

//// display(200);

// for(i=0;i<5;i++)

// DisplayOneChar(i,1,shuju[i]); //显示字库中的中文数子

tran2( );

// Delay400Ms();

// Delay400Ms();//通过延时防止因csb1接收引起中断

tran3( );

// Delay400Ms();

// Delay400Ms();//通过延时防止因csb1接收引起中断

tran4( );

Delay400Ms();

Delay400Ms();

// Delay400Ms();

// Delay400Ms();//通过延时防止因csb1接收引起中断

// LCDClear();//清屏

// Delay400Ms();

// LCDClear();

}

}

////////////////////////////////////////////////////

//延时函数体

void delay100us()

{

uchar i;

for(i=0;i<50;i++)

;

}

void delay_nms(uint n)

{

uchar i;

while(n--)

{

for(i=123;i>0;i--);

}

}

//////超声波发射程序

void tran1(void)///可通过设置K数值选择执行相应的I0发射超声波

{

uchar i;

float temp;

TMOD=0XF0;

TMOD |=0X01;

TL0=0X47;

TH0=0XFF;

TR0=0;

ET0=0;//关闭timme0中断?

csb1=1;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

csb1=0;

TR0=0;

TH0=0;

TL0=0;

while(in1==0);

TR0=1;

while(in1==1);

TR0=0;

temp=(TH0*256+TL0)*1.08/58; //1.08为分频之后的数

//蜂鸣器响

if(temp<20)beef=0;

delay_nms(60);

beef=1;

//

if(temp>4000)temp=4000;

dis =(unsigned int)temp;//dis将直接用于显示函数

delay_nms(60);

display(dis);

for(i=0;i<6;i++)

DisplayOneChar(i,1,shuju[i]); //显示字库中的中文数子

}

void tran2(void)///可通过设置K数值选择执行相应的I0发射超声波

{

uchar i;

float temp;

TMOD=0XF0;

TMOD |=0X01;

TL0=0X47;

TH0=0XFF;

TR0=0;

ET0=0;//关闭timme0中断?

csb2=1;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

csb2=0;

TR0=0;

TH0=0;

TL0=0;

while(in2==0);

TR0=1;

while(in2==1);

TR0=0;

temp=(TH0*256+TL0)*1.08/58; //1.08为分频之后的数

//蜂鸣器响

if(temp<20)beef=0;

delay_nms(60);

beef=1;

//

if(temp>4000)temp=4000;

dis =(unsigned int)temp;//dis将直接用于显示函数

delay_nms(60);

display(dis);

shuju[0]=shuju[0]+1;

for(i=0;i<6;i++)

DisplayOneChar(i,2,shuju[i]); //显示字库中的中文数子

}

void tran3(void)///可通过设置K数值选择执行相应的I0发射超声波

{

uchar i;

float temp;

TMOD=0XF0;

TMOD |=0X01;

TL0=0X47;

TH0=0XFF;

TR0=0;

ET0=0;//关闭timme0中断?

csb3=1;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

csb3=0;

TR0=0;

TH0=0;

TL0=0;

while(in3==0);

TR0=1;

while(in3==1);

TR0=0;

temp=(TH0*256+TL0)*1.08/58; //1.08为分频之后的数

//蜂鸣器响

if(temp<20)beef=0;

delay_nms(60);

beef=1;

//

if(temp>4000)temp=4000;

dis =(unsigned int)temp;//dis将直接用于显示函数

delay_nms(60);

display(dis);

shuju[0]=shuju[0]+2;

for(i=0;i<6;i++)

DisplayOneChar(i,3,shuju[i]); //显示字库中的中文数子

}

void tran4(void)///可通过设置K数值选择执行相应的I0发射超声波

{

uchar i;

float temp;

TMOD=0XF0;

TMOD |=0X01;

TL0=0X47;

TH0=0XFF;

TR0=0;

ET0=0;//关闭timme0中断?

csb4=1;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

nop;

csb4=0;

TR0=0;

TH0=0;

TL0=0;

while(in4==0);

TR0=1;

while(in4==1);

TR0=0;

temp=(TH0*256+TL0)*1.08/58; //1.08为分频之后的数

//蜂鸣器响

if(temp<20)beef=0;

delay_nms(60);

beef=1;

//

if(temp>4000)temp=4000;

dis =(unsigned int)temp;//dis将直接用于显示函数

delay_nms(60);

display(dis);

shuju[0]=shuju[0]+3;

for(i=0;i<6;i++)

DisplayOneChar(i,4,shuju[i]); //显示字库中的中文数子

}

///////////////////////////////////

void display(uint dat)

{

uchar i,j,k;//定义变量

i=dat/100;//白

j=dat0/10;//十位

k=dat0;//个

shuju[0]=65;

shuju[1]=i+48;

shuju[2]='.';

shuju[3]=j+48;

shuju[4]=k+48;

shuju[5]=109;//或者shuju[4]='m';

}

////////////////////////////////////////////

//1写数据

void WriteDataLCD(unsigned char WDLCD)

{

ReadStatusLCD(); //检测忙

LCD_RS = 1;

LCD_RW = 0;

LCD_Data = WDLCD;

LCD_E = 1;

LCD_E = 1;

LCD_E = 1;

LCD_E = 0;

}

//2写指令

void WriteCommandLCD(unsigned char WCLCD,BuysC) //BuysC为0时忽略忙检测

{

if (BuysC) ReadStatusLCD(); //根据需要检测忙

LCD_RS = 0;

LCD_RW = 0;

LCD_Data = WCLCD;

LCD_E = 1;

LCD_E = 1;

LCD_E = 1;

LCD_E = 0;

}

//3读数据

unsigned char ReadDataLCD(void)

{

LCD_RS = 1;

LCD_RW = 1;

LCD_E = 0;

LCD_E = 0;

LCD_E = 1;

return(LCD_Data);

}

//4读状态

unsigned char ReadStatusLCD(void)

{

LCD_Data = 0xFF;

LCD_RS = 0;

LCD_RW = 1;

LCD_E = 1;

while (LCD_Data & Busy); //检测忙信号

LCD_E = 0;

return(LCD_Data);

}

void LCDInit(void) //5LCM初始化

{

WriteCommandLCD(0x30,1); //显示模式设置,开始要求每次检测忙信号

WriteCommandLCD(0x01,1); //显示清屏

WriteCo

声明: 本文转载自其它媒体或授权刊载,目的在于信息传递,并不代表本站赞同其观点和对其真实性负责,如有新闻稿件和图片作品的内容、版权以及其它问题的,请联系我们及时删除。(联系我们,邮箱:evan.li@aspencore.com )
0
评论
  • 【下载】电源设计工程师指南(共542页)


  • 相关技术文库
  • 单片机
  • 嵌入式
  • MCU
  • STM
  • 51系列单片机的引脚图

    51系列单片机的引脚图 端子介绍 l P0.0~P0.7 P0口8位双向口线(在引脚的39~32号端子)。 l P1.0~P1.7 P1口8位双向口线(在引脚的1~8号端子)。 l P2.0~P2.7 P2口8位双向口线(在引脚的21~28号端子)。 l P3.0~P3.7 P2口8...

    5小时前
  • 51单片机串口通信需要加超时中断吗?

    接收数据时,超过一定时间就算出错. 这个超时的时间是单片机自己算出的吗?超时的时间是由编程序的人定的,他定多长就多长从一段程序开始 实现电脑向 单片机发送一些数据,单片机返回Iget +数据 #include #define u...

    5小时前
  • 51单片机电机pid控制系统程序

    51单片机电机pid控制系统程序 本程序来源网上,小编进行了简单修改,还未验证。 /************************* 应用背景:直流电机的额定功率12V,额定转速3000rpm,光码盘12孔, 晶振为12MHz,定时/计数器T0检测转...

    5小时前
  • 51单片机指令集设计实例

    当你编写完一个程序后,通过软件调试,你可以看到程序的指令集,从指令集你就能知道单片机内部是如何工作的,比如数据存取的寄存器与地址,下面是电工之家找的一篇指令集,讲的蛮好的。 一、数据传送类指令(7种助...

    5小时前
  • 51单片机定时器工作原理及用法

    TMOD : 控制定时器的工作方式。8个bit,高四位 bit 控制 T1,、低四位 bit 控制 T0。因为定时器有4种工作方式;TMOD = 0x00(工作方式0),TMOD = 0x01(工作方式0),TMOD = 0x02(工作方式2),TMOD = 0x03(工作方式3)。...

    5小时前
  • 51单片机指令系统寻址方式

    指令给出参与运算的数据方式称为寻址方式。换句话说,寻址方式就是寻找确定参与操作的数的真正地址。 在MCS-51系列单片机的指令系统中寻址方式共有7种,立即寻址方式,mcs-51单片机的一种寻址方式,操作数就写在指...

    5小时前
  • 51单片机学习单片机之路总结

    学习单片机有一学期了,现在也由51转到STM32了。一直想对51的学习做一个总结。也希望对别人有一些启发。也给后学者提供一些建议。当然本文是我对自己学习过程的总结,若有不对的地方,还请高手指出。 我想,再看本...

    5小时前
  • 入手STM32单片机的知识点总结

    从STM32新建工程、编译下载程序出发,让新手由浅入深,尽享STM32标准库开发的乐趣

    前天
  • 单片机定时器之改良版:时间轮定时器

    前段时间把自己以前用的单片机定时器整理出来,我称之为简单定时器,这种简单定时器比较适合定时器使用量少的程序中,如果定时器数量要求多,精度要求高,效率就会有问题,为此,俺就实现了一个时间轮定时器,简单测...

    前天
  • 详聊单片机BootLoader的发展进化史

    Bootloader是一个用于引导嵌入式系统的软件程序,通常存储在芯片的非易失性存储器中,如闪存。Bootloader的主要功能是在系统上电后,负责初始化系统硬件并加载操作系统或应用程序到内存中执行。

    06-18
下载排行榜
更多
评测报告
更多
广告