/*USART1config1152008-N-1*/
USART1_Config();
Distance_Config();//距离转换函数,在上面的说明中有提到,但是具体的实现和上面的公式是不相同的
CLI();//关闭总中断
SEI();//开启总中断
Tim3_Config();//定时器初始化
GPIO_ResetBits(GPIOA,GPIO_Pin_5);//先拉低电平
while(1)
{
GPIO_SetBits(GPIOA,GPIO_Pin_5);//再拉高电平,这里拉低拉高电平是根据上面给出的测距原理来写的
Delay(30);//延时30个us,注意这里使用的是粗略的延时函数,测距原理中说是延时10个us,这里给30个us也无妨
//Delay(20);
//Delay(20);
GPIO_ResetBits(GPIOA,GPIO_Pin_5);
TIM3-》CNT=0;//TIM3的计数器清零
while(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4)==0);//等待ECHO的高电平
TIM_Cmd(TIM3,ENABLE);//运行TIM3进行计数
while((GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_4)==1)&&(TIM3-》CNTARR-10));
TIM_Cmd(TIM3,DISABLE);
count=TIM3-》CNT;
printf(“count=%d”,count);
distance=ChangeDistance(count);
printf(“μ±?°?àà??a£o%fn”,distance);
Delay(20000);
Delay(20000);//这里的延时没有具体的意义,可以去掉
while(1);//程序测距一次后卡死在这里
}
}
距离转换函数
floatChangeDistance(unsignedintcout1)
{
floatdistance=0;
printf(“cou1=%dn”,cout1);
distance=cout1/58.0;
returndistance;
}
参数是TIM3-》CNT的计数,也就是高电平的时间,distance是测距距离
注意:distance的单位是厘米
关于转换的公式为什么是cout1/58.0 这里我也不是很清楚
另外:当ECHO引脚输出高电平后,另一种思路是采用上升沿触发定时器中断的方式来计算高电平时间的,我认为没有必要再配置一次中断,使用while()来等待这个高电平的方法完全可以。当然不容忽视的一个问题是,如果始终检测不到高电平,程序会卡死在while()语句这里,因此有必要加上一个时间的判断。
通过串口打印相应的测量结果。
注意:1 有的超声波模块是有温度校准的,有温度校准的模块测距精度要高一些。
2 模块再稳定下来之后测距是比较准确的,即便是我上面写的代码,也需要稳定一下再测距才可以,举个例子,当你将超声波模块对着墙时,刚上电测到的距离并不准确,但是很短时间后测距就会变得非常精准。
3 上面的程序采用的是电平触发的方式,我看到有的说使用串口的方式会使精确度更高,但我没有做相关的实验。
4 想要提高测距精度的另一种方法就是多次测量,去掉最大值,最小值后取平均值的方法,我在另一份程序中采用的是测五次距离,然后取出平均值得方法,需要注意多次测量的周期最好大于60ms。