蓝桥杯第八届国赛-----超声波测距机

目录

题目

考察要求

1、超声波模块

2、独立按键,LED,数码管

3、eeprom

4、DAC

代码

1、main

2、iic

3、iic.h

总结


题目

考察要求

1、超声波模块

        超声波发送和接收

void Send_wacv()
{
	uchar i=8;
	do{
		TX=1;
		semopon;semopon;semopon;semopon;semopon;semopon;semopon;semopon;
		TX=0;
		semopon;semopon;semopon;semopon;semopon;semopon;semopon;semopon;
	}while(i--);
}

uint Read_DIST()
{
	if(dist_flag)
	{
	  dist_flag=0;
	  Send_wacv();
	  TR1=1;
	  while((RX==1)&&(TF1==0));
	  TR1=0;
	  if(TF1==1)
	  {
	  	 TF1=0;
		 ucDist=999;//999-256*3=231  256是eeprom最大存储容量
	  }else{
	  	   t=TH1;
		   t<<=8;
		   t|=TL1;
		   ucDist=(int)(t*0.017);
	  }
	}
	TH1=0;
	TL1=0;
	return ucDist;
}

2、独立按键,LED,数码管

        不说了,每次都有,提下小灯L1的写法:L1是以一秒为间隔闪烁,闪烁三次后熄灭。可以定义一个变量让它每一秒翻转一次实现小灯闪烁,等到它为5秒时置为0就将其关闭小灯。

//在定时器服务函数写这个	

    led_count++;
	if(led_count%1000==0)
	{
	  led_flag=~led_flag;
	}

//小灯函数里面写
if(flag==1&&led_count<6000)   //在保存的情况下间隔1s亮灭3次
	{	 
		if(led_flag){		
			LED|=1;
			}
		else{
			LED&=~1;
		}
	}
	else {
		led_count=0;
		flag=0;
		LED&=~1;	
	}

3、eeprom

4、DAC

        这两个一起写在iic里面,注意写的时候加延时,因为一直连续写会导致数据错误,读可以不用加延时,也可以加,看你个人。DAC的话注意它是存256个字节。比如我们将999存入eeprom里面,实际存入的只是231,999-256*3=231。读取出来也只是231。还有就是中断记关了,记得开


//Write eeprom
void Write_eeprom(unsigned char add,unsigned char dat)
{
   IIC_Start();
   IIC_SendByte(0xa0);
   IIC_WaitAck();
   IIC_SendByte(add);
   IIC_WaitAck();
   IIC_SendByte(dat);
   IIC_WaitAck();
   IIC_Stop();
}

unsigned char Read_eeprom(unsigned char add)
{
	unsigned char temp;
	EA=0;
   IIC_Start();
   IIC_SendByte(0xa0);
   IIC_WaitAck();
   IIC_SendByte(add);
   IIC_WaitAck();

   IIC_Start();
   IIC_SendByte(0xa1);
   IIC_WaitAck();
   temp=IIC_RecByte();
   IIC_WaitAck();
   IIC_Stop();
   EA=1;
   return temp;
}

//Write DAC
void Write_DAC(unsigned char dat)
{
   IIC_Start();
   IIC_SendByte(0x90);
   IIC_WaitAck();
   IIC_SendByte(0X40);
   IIC_WaitAck();
   IIC_SendByte(dat);
   IIC_WaitAck();
   IIC_Stop();
}

代码

1、main

#include<stc15f2k60s2.h>
#include<intrins.h>
#include"iic.h"

#define uchar unsigned char
#define uint unsigned int
#define semopon {_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();_nop_();}
sbit Buzzer = P0^6;
sbit Relay =P0^4;
//超声波
sbit TX = P1^0;
sbit RX = P1^1;

/*************  本地常量声明    **************/
uchar code t_display[]={                       //标准字库
//   0    1    2    3    4    5    6    7    8    9    A    B    C    D    E    F
    0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71,
//black  -     H    J    K    L    N    o   P    U     t    G    Q    r   M    y
    0x00,0x40,0x76,0x1E,0x70,0x38,0x37,0x5C,0x73,0x3E,0x78,0x3d,0x67,0x50,0x37,0x6e,
    0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x46};    //0. 1. 2. 3. 4. 5. 6. 7. 8. 9. -1

uchar code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};      //位码
uchar table[]={16,16,16,16,16,16,16,16};
uchar ReadData,Trg,Cont;
bit key_flag;
uchar LED;

//超声波测距
uint t,ucDist,distance;
bit dist_flag;


//功能
uchar new_distance,old_distance;//本次测量距离和上次测量距离
uchar set_mode;//0为测量距离界面,1为回显界面,2为盲区
uchar eep[4];  //存入测量结果
uchar number;//盲区值
uchar eep_count;//
uchar bh=1;//回显结果编号
bit flag;//测量标志
bit led_flag;
uchar count=0;
uint led_count;
void Delay10ms()		//@12.000MHz
{
	unsigned char i, j;

	i = 117;
	j = 184;
	do
	{
		while (--j);
	} while (--i);
}

//关闭外设
void allinit()
{
	P2=0xa0;Buzzer = 0; Relay = 0;P2=0x00;
	P2=0x80;P0=0xff;P2=0x00;
}

void Set_LED()
{
	P2=0x80;
	P0=~LED;
	P2=0x00;
}

void Timer0Init(void)		//1毫秒@12.000MHz
{
	EA=1;ET0=1;
	AUXR |= 0x80;		//定时器时钟1T模式
	TMOD &= 0xF0;		//设置定时器模式
	TL0 = 0x20;		//设置定时初值
	TH0 = 0xD1;		//设置定时初值
	TF0 = 0;		//清除TF0标志
	TR0 = 1;		//定时器0开始计时
}

void Timer0() interrupt 1
{
	static uint smg_count,key_count,i,dist_count;
	if(++smg_count>=1)
	{
	   smg_count=0;
	   P2=0xe0;P0=0xff;P2=0x00;
	   P2=0xc0;P0=T_COM[i];P2=0x00;
	   P2=0xe0;P0=~t_display[table[i]];P2=0x00;
	   i++;
	   if(i>=8)i=0;
	}
	if(++key_count>=10)
	{
	  key_count=0;
	  key_flag=1;
	}
	if(++dist_count>=200)
	{
	  dist_count=0;
	  dist_flag=1;
	}
	led_count++;
	if(led_count%1000==0)
	{
	  led_flag=~led_flag;
	}
}
void Send_wacv()
{
	uchar i=8;
	do{
		TX=1;
		semopon;semopon;semopon;semopon;semopon;semopon;semopon;semopon;
		TX=0;
		semopon;semopon;semopon;semopon;semopon;semopon;semopon;semopon;
	}while(i--);
}

uint Read_DIST()
{
	if(dist_flag)
	{
	  dist_flag=0;
	  Send_wacv();
	  TR1=1;
	  while((RX==1)&&(TF1==0));
	  TR1=0;
	  if(TF1==1)
	  {
	  	 TF1=0;
		 ucDist=999;//999-256*3=231  255是eeprom最大存储容量
	  }else{
	  	   t=TH1;
		   t<<=8;
		   t|=TL1;
		   ucDist=(int)(t*0.017);
	  }
	}
	TH1=0;
	TL1=0;
	return ucDist;
}
void  Write_eep_dist()
{
	switch(eep_count)
	{
		case 0:
			 Write_eeprom(0x01,distance);Delay10ms();
			 new_distance=Read_eeprom(0x01); Delay10ms();
			 old_distance=Read_eeprom(0x04);  Delay10ms();
		break;
		case 1:
		  	 Write_eeprom(0x02,distance);Delay10ms();
			 new_distance=Read_eeprom(0x02); Delay10ms();
			 old_distance=Read_eeprom(0x01);
		break;
		
		case 2:
		   	 Write_eeprom(0x03,distance);Delay10ms();
			 new_distance=Read_eeprom(0x03); Delay10ms();
			 old_distance=Read_eeprom(0x02);
		break;
		case 3:
		   	 Write_eeprom(0x04,distance);Delay10ms();
			 new_distance=Read_eeprom(0x04); Delay10ms();
			 old_distance=Read_eeprom(0x03); Delay10ms();
		break;
	}
}



void ReadKey()
{
	ReadData=P3^0xff;
	Trg=ReadData&(ReadData^Cont);
	Cont=ReadData;
}
void disp_key()
{
	if(key_flag)
	{
		key_flag=0;
		Set_LED();
		ReadKey();
		switch(Trg)
		{
			case 0x08:
			  set_mode=0;
			  distance=Read_DIST();
			  Write_eep_dist();		   //存入测量结果
			  eep_count++;
			  if(eep_count>=4)eep_count=0;
			  flag=1;
		break;
			case 0x04:
			 if(set_mode==1)
			 {
			   set_mode=0;
			 }else{
			   set_mode=1;
			 }
		break;
			case 0x02:
			  if(set_mode==2)
			 {
			 	Write_eeprom(0x06,number);
				Delay10ms();
			   set_mode=0;
			 }else{
			   set_mode=2;
			 }
		break;
			case 0x01:
			if(set_mode==1)
			{
			  bh++;
			  if(bh>=5)bh=1;
			}else if(set_mode==2)
			{
			  number+=10;
			  if(number>=40)number=0;
			}
		break;	
		}
	}
}

void disp_DAC()
{
	if(distance<=number)
	{
	  Write_DAC(0);
	}
	else if(distance>number)
	{
		if((int)((distance-number)*0.2)>5)
		{
	 		Write_DAC(255);
		}else{
			Write_DAC((int)(((distance-number)*0.2)*255/5.0));	
		}
	}
}

void disp_LED()
{
	uchar i=0;
	if(set_mode==2)
	{
	  LED|=64;
	}else{
		LED&=~64;
	}
	if(set_mode==1)
	{
	  LED|=128;
	}else{
		LED&=~128;LED&=~2;LED&=~4;LED&=~8;LED&=~16;LED&=~32;
	}
	if(flag==1&&led_count<6000)   //在保存的情况下间隔1s亮灭3次
	{	 
		if(led_flag){		
			LED|=1;
			}
		else{
			LED&=~1;
		}
	}
	else {
		led_count=0;
		flag=0;
		LED&=~1;	
	}
}

void  main()
{
	allinit();
	Timer0Init();
	EA=0;
	number=Read_eeprom(0x06);
	EA=1;
	while(1)
	{
	  disp_DAC();
	  switch(set_mode)
	  {
	  	 	case 0:
			 table[0]=12;table[1]=16;
			 table[2]=distance/100;
			 table[3]=distance/10%10;
			 table[4]=distance%10;
			 table[5]=old_distance/100;
			 table[6]=old_distance/10%10;
			 table[7]=old_distance%10;
		break;
			case 1:
			 EA=0;
			 eep[0]=Read_eeprom(0x01);Delay10ms();eep[1]=Read_eeprom(0x02);Delay10ms();
			 eep[2]=Read_eeprom(0x03);Delay10ms();eep[3]=Read_eeprom(0x04);Delay10ms();
			 EA=1;
			 table[0]=bh;table[1]=16;table[2]=16;table[3]=16;table[4]=16;
			 table[5]=eep[bh-1]/100;
			 table[6]=eep[bh-1]/10%10;
			 table[7]=eep[bh-1]%10;
		break;
	  		case 2:
			 table[0]=15;table[1]=16;table[2]=16;table[3]=16;
	   		 table[4]=16;table[5]=16;table[6]=number/10%10;table[7]=number%10;
		break;
	  }
	  disp_LED();
	  disp_key();
	}
}

2、iic

/*
  程序说明: IIC总线驱动程序
  软件环境: Keil uVision 4.10 
  硬件环境: CT107单片机综合实训平台 8051,12MHz
  日    期: 2011-8-9
*/

#include "reg52.h"
#include "intrins.h"

#define DELAY_TIME 5

#define SlaveAddrW 0xA0
#define SlaveAddrR 0xA1

//总线引脚定义
sbit SDA = P2^1;  /* 数据线 */
sbit SCL = P2^0;  /* 时钟线 */

void IIC_Delay(unsigned char i)
{
    do{_nop_();}
    while(i--);        
}
//总线启动条件
void IIC_Start(void)
{
    SDA = 1;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 0;
    IIC_Delay(DELAY_TIME);
    SCL = 0;	
}

//总线停止条件
void IIC_Stop(void)
{
    SDA = 0;
    SCL = 1;
    IIC_Delay(DELAY_TIME);
    SDA = 1;
    IIC_Delay(DELAY_TIME);
}

//等待应答
bit IIC_WaitAck(void)
{
    bit ackbit;
	
    SCL  = 1;
    IIC_Delay(DELAY_TIME);
    ackbit = SDA;
    SCL = 0;
    IIC_Delay(DELAY_TIME);
    return ackbit;
}

//通过I2C总线发送数据
void IIC_SendByte(unsigned char byt)
{
    unsigned char i;

    for(i=0; i<8; i++)
    {
        SCL  = 0;
        IIC_Delay(DELAY_TIME);
        if(byt & 0x80) SDA  = 1;
        else SDA  = 0;
        IIC_Delay(DELAY_TIME);
        SCL = 1;
        byt <<= 1;
        IIC_Delay(DELAY_TIME);
    }
    SCL  = 0;  
}

//从I2C总线上接收数据
unsigned char IIC_RecByte(void)
{
    unsigned char i, da;
    for(i=0; i<8; i++)
    {   
    	SCL = 1;
	IIC_Delay(DELAY_TIME);
	da <<= 1;
	if(SDA) da |= 1;
	SCL = 0;
	IIC_Delay(DELAY_TIME);
    }
    return da;    
}

//Write eeprom
void Write_eeprom(unsigned char add,unsigned char dat)
{
   IIC_Start();
   IIC_SendByte(0xa0);
   IIC_WaitAck();
   IIC_SendByte(add);
   IIC_WaitAck();
   IIC_SendByte(dat);
   IIC_WaitAck();
   IIC_Stop();
}

unsigned char Read_eeprom(unsigned char add)
{
	unsigned char temp;
	EA=0;
   IIC_Start();
   IIC_SendByte(0xa0);
   IIC_WaitAck();
   IIC_SendByte(add);
   IIC_WaitAck();

   IIC_Start();
   IIC_SendByte(0xa1);
   IIC_WaitAck();
   temp=IIC_RecByte();
   IIC_WaitAck();
   IIC_Stop();
   EA=1;
   return temp;
}

//Write DAC
void Write_DAC(unsigned char dat)
{
   IIC_Start();
   IIC_SendByte(0x90);
   IIC_WaitAck();
   IIC_SendByte(0X40);
   IIC_WaitAck();
   IIC_SendByte(dat);
   IIC_WaitAck();
   IIC_Stop();
}

3、iic.h

#ifndef _IIC_H
#define _IIC_H

void IIC_Start(void); 
void IIC_Stop(void);  
bit IIC_WaitAck(void);  
void IIC_SendByte(unsigned char byt); 
unsigned char IIC_RecByte(void); 
//Write eeprom
void Write_eeprom(unsigned char add,unsigned char dat);
unsigned char Read_eeprom(unsigned char add);
//Write DAC
void Write_DAC(unsigned char dat);
#endif

总结

        一共刷了三次国赛题,才发现还是太菜了,每次都没能在规定时间内完成该有的功能,唯独只有第十一届的国赛题完成了,还是得大量的敲代码,要不然代码思维真的会退步。

        希望本篇博客对你有帮助,有什么不足的,请您指正,一起学习!最近在学习stm32,遇到了很多坑,后面我会给大家发一些,免得大家后面学习stm32时踩坑,浪费时间。

        还有就是那些要钱的博客有些真的很坑!千万别踩坑,我朋友就是例子。别用钱去买无用的知识。

  • 6
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值