2019硬件组答辩作品----基于STC89C52的智能小船

2019硬件组答辩作品----基于STC89C52的智能小船

主要功能简介:

作品基本功能为遥控小船,主要通过红外遥控按键控制小船前进后退以及转向,其次可通过超声波测距靠岸自动停止,并在LCD1602液晶屏显示超声波测距数值和当前状态,最后以舵机和直流电机提供前进以及转向的动力。

功能介绍:

1.LCD1602液晶屏显示
2.L298N控制直流电机
3.红外遥控遥控小船
4.超声波测距及靠岸停止
5.舵机摆动提供动力

作品功能实现基本要求:要保证小船能在水面上浮起来,及需要尽量降低小船整体质量,从而保证小船的通电部分不会接触到水。

模块介绍

1.LCD1602模块
在这里插入图片描述

遇到的问题:因为需要保证减轻小船质量的大前提,所以小船选择的是最小系统板。然而用最小系统板接线都是其次,主要是最小系统板没有滑动变阻器,导致液晶屏亮度过高。在寻找合适电阻无果后发现用二极管刚好合适。

2.L298N模块
L298N可驱动2路直流电机,根据向IN1,IN2,IN3,IN4输入不同的逻辑信号来控制电机正转反转和停止,但必须是在使能端ENA,ENB为高电平时才有效。即当使能信号为零时,电机处于制动状态。
在这里插入图片描述
调速:利用使能端ENA,ENB输入一个pwm信号便能实现对直流电机调速
遇到的问题:可能是工作室的电池比较旧吧,第一次用工作室四节电池的电池盒接L298N,结果模块不能工作,最后用六节电池才勉强能转动起来。

3.红外遥控模
在这里插入图片描述
NEC协

  1. 8 位地址码,8 位命令码
  2. 完整发射两次地址码和命令码,以提高可靠性
  3. 脉冲时间长短调制方式
  4. 38KHz 载波频率
  5. 位时间 1.12ms (0)或 2.25ms(1)

在这里插入图片描述
遇到的问题: 一开始不明白为什么需要超时退出,但是红外信号在传输的时候时有可能会出错的,所以超时退出是有必要的。其次就是红外的数据用完是需要清零,之前想用红外的数据等于对应键值,控制位就取反来控制超声波模块的开关。但是就是应为红外的数据没有清零,结果当键按下时,控制位会不断取反,导致不能成功控制超声波模块的开关。

4.超声波测距模块
在这里插入图片描述
(1)采用IO口TRIG触发测距,给最少10us的高电平信号。
(2)模块自动发送8个40khz的方波,自动检测是否有信号返回
(3)有信号返回,通过IO口ECHO输出一个高电平,高电平持续的时间就是超声波从发射到返回的时间。测试距离=(高电平时间*声速(340M/S))/2

遇到的问题:因为舵机和超声波测距用了两个定时器,然后代码又是分模块写,于是舵机初始化中TMOD=0x10将超声波测距初始化的TMOD=0x10覆盖,导致超声波测距不准。

5.SG90舵机模块
在这里插入图片描述
舵机是利用脉冲调制方法来控制角度,即旋转角度的大小是由来自控制线的持续的脉冲的长度所控制的。舵机的控制信号为周期是20ms的脉宽调制(PWM)信号,其中脉冲宽度从0.5ms-2.5ms,相对应舵盘的位置为0-180度,呈线性变化。
而pwm波进入内部电路便会产生一个偏置电压,触发电机通过减速齿轮带动电位器移动,使电压差为零时,电机停转,从而达到伺服的效果。

遇到的问题:因为用混合的旧电池拼的电池盒给最小系统板供电,电压不稳定导致舵机抽搐,甚至使整个小船卡死。

需要改进的地方:

红外遥控不灵敏,不能灵活地操纵小船;
舵机存在抽搐的情况,且抽搐有时还会导致单片机卡死;
转弯时动力不足,转向很慢。

收获:

1.以上模块的模块的用法,51单片机的知识
2.自己动手实践做东西的经历
3.如何自己发现问题解决问题

另附小船实物图一张:

下面为小船的源代码:

#include<reg52.h>
#include<intrins.h>

#define uint unsigned int
#define uchar unsigned char

#define LCD_DA P0
sbit IN1=P1^0;
sbit IN2=P1^1;
sbit IN3=P1^2;
sbit IN4=P1^3;
sbit pr=P2^3;	//L298N

sbit signal=P3^2;//红外

sbit LCD_RS=P2^6;
sbit LCD_RW=P2^5;
sbit LCD_E=P2^7;
sbit LCD_switch=P2^2; //LCD1602

sbit pwm=P2^4; //  舵机

sbit Trig = P2^1;
sbit Echo = P2^0;  //超声波测距

uchar dis[]="Distance : ";
uchar dir[][11]={"go ahead  ","go back   ","  stop    ","turn left ","turn right","state:"};
uchar a[]="      ";
uchar asc[]={'0','1','2','3','4','5','6','7','8','9','.','-','M'};
uchar flag=0;
uchar infdata[4];
uchar spr;
uchar dirdata;
uchar dj;
uchar count;
uchar count2;
uchar ser;
uchar wave_switch;


//延时函数
/****************************************/
void delay(uint time){
	uchar k;
	while(time--)
		for(k=111;k>0;k--);	 //延迟1ms
}




//LCD1602
/****************************************/
uchar busy()//不忙检测
{
	uchar sta;
	LCD_RS=0;
	LCD_DA=0xff;//读数据之前都赋高电平
	LCD_RW=1;
	LCD_E=1;
	sta=LCD_DA;
	LCD_E=0;
	return sta&0x80;
}

void wrt_com(uchar lcddat)
{
	while(busy());
	LCD_RS=0;
	LCD_RW=0;
	LCD_DA=lcddat;
	LCD_E=1;
	LCD_E=0;
}

void wrt_dat(uchar lcddat)
{
	 while(busy());
	LCD_RS=1;
	LCD_RW=0;
	LCD_DA=lcddat;
	LCD_E=1;
	LCD_E=0;
}

void display(uchar x,uchar y,uchar lcddat)
{
	if(y)
		x|=0x80;
	else
		x=x|0x80|0x40;
	wrt_com(x);
	wrt_dat(lcddat);
}

 void lcd_init()
{
	LCD_switch=0;
	wrt_com(0x38);
	wrt_com(0x0c);
	wrt_com(0x06);
	wrt_com(0x01);
}






//驱动模块
/****************************************/
void motinit()
{
	IN1=0;
	IN2=0;
	IN3=0;
	IN4=0;
	spr=50;
	dirdata=0x1c;	
}

void spreed()//调速
{
	switch(infdata[2])
	{
		case 0x0d:spr=50;count2=0;break;// 200+
		case 0x19:spr=35;count2=0;break;// 100+
		case 0x16:spr=20;count2=0;break;// 0
		default:;
	}
}

void direction(uchar i)
{
	 uchar j,k,n;
	switch(i)
	{	 												 
		case 0x18:IN1=0;IN2=1;IN3=0;IN4=1;k=0;break; //前进	   2
		case 0x52:IN1=1;IN2=0;IN3=1;IN4=0;k=1;break; //后退    8
		case 0x1c:IN1=0;IN2=0;IN3=0;IN4=0;k=2;break; //停止    5
		case 0x5a:IN1=0;IN2=1;IN3=1;IN4=0;k=3;dj=4;break; //左转    6
		case 0x08:IN1=1;IN2=0;IN3=0;IN4=1;k=4;dj=2;break; //右转    4
		default:;
	}

	 for(j=0;dir[5][j];j++)
		display(j,0,dir[5][j]);
	 for(n=0;dir[k][n];n++)
		display(j++,0,dir[k][n]);
			
}







//红外线模块
/****************************************/
void infrayinit()
{
	IT0=1;
	EX0=1;
	EA=1;	
	signal=1;
}

void receive() interrupt 0
{
	uchar time,i,j;
	uint err;
	delay(7);
	if(!signal)			//?
	{
		err=300;
		while(!signal&&err>0)//等待起始码低电平结束
		{
			err--;
		}
		 if(signal)
		 {
		 	err=500;
		 	while(signal&&err>0)//等待1起始码高电平结束
			{
				err--;
			}		
			for(i=0;i<4;i++)//开始收集数据
			{
				for(j=0;j<8;j++)
				{
					time=0;
					err=60;
					while(!signal&&err>0)//等待数据码低电平结束
					{
						err--;
					}
					err=200;
					while(signal&&err>0)//等待数据码高电平结束
					{
						err--;
						time++;
					   if(time>100)
						return;
					}
					infdata[i]>>=1;
					if(time>50)
					{
						infdata[i]|=0x80;
					}
				}
			}
			if(infdata[2]!=~infdata[3])//数据码验证
				for(i=0;i<4;i++)
					infdata[i]&=0x00;
		
		 }
	}
	

}

void infray()
{
	if(infdata[2]==0x5a||infdata[2]==0x18||infdata[2]==0x1c||infdata[2]==0x08||infdata[2]==0x52) //取驱动方向数据
	{
		dirdata=infdata[2];
		infdata[2]=0x00;
	}
	if(infdata[2]==0x09) //液晶屏开关  EQ
	{
		LCD_switch=~LCD_switch;
		 infdata[2]=0x00;
	}
	if(infdata[2]==0x46)	//超声波测距开关   CH
	{
		wave_switch=~wave_switch;
			 infdata[2]=0x00;	
	}
	if(infdata[2]==0x43) //	舵机开关  >|
	{
		ser=~ser;
		infdata[2]=0x00;
	}
}
 








//超声波测距
/****************************************/
void wave_init()
{
	TMOD=0x01;
	ET0=1;
	TH0=0;
	TL0=0;
	TR0=0;
	EA=1;
	wave_switch=0;
}

void wave_count()
{	 
	uint s,i;
	s=(1.7*(TH0*256+TL0))/100;	  //cm
	TH0=0;
	TL0=0;
	for(i=0;dis[i];i++)
		display(i,1,dis[i]);
	if(s>700||flag)
	{
		flag=0;
		display(i++,1,asc[11]);
		display(i++,1,asc[10]);
		display(i++,1,asc[11]);
		display(i++,1,asc[11]);
		display(i++,1,asc[12]);
	}
	else
	{
		display(i++,1,asc[s/100]);
		display(i++,1,asc[10]);
		display(i++,1,asc[s%100/10]);
		display(i++,1,asc[s%10]);
		display(i++,1,asc[12]);
	}
  	if(s<20)//超声波停止
	{
		dirdata=0x1c;
	}	
}

void startwave()
{
	uchar i;
	Trig=1;
	for(i=0;i<21;i++)
	{
		_nop_();
	}
	Trig=0;
}

void wave()
{
	uchar i;
	startwave();
	while(!Echo);
	TR0=1;
	while(Echo);
	TR0=0;
	if(wave_switch)
		wave_count();
	else
	{
		TH0=0;
		TL0=0;
	   for(i=0;a[i];i++)
	   display(i+11,1,a[i]);
	}	 
	delay(100);
}

void timer() interrupt 1
{
	flag=1;
}



//舵机模块
/****************************************/
void servoinit()
{
	EA=1;
	ET1=1;
	TMOD|=0x10;
	TR1=1;
	TH1=0xfe;
	TL1=0x33;
	dj=3;
	count=0;
	count2=0;
	ser=0;
}

void sertime() interrupt 3
{
	TH1=0xfe;
	TL1=0x33;
	if(count<dj)
		pwm=1;
	else
		pwm=0;

	if(count2<spr)
		pr=1;	
	else
		pr=0;
						
		count++;
		count%=40;
		count2++;
		count2%=50;	
}

void servo()
{

	if(ser||dirdata==0x18)
	{
		dj=2; count=0;
		delay(300);
		dj=4; count=0;
		delay(300);
	}
	else
	{
		dj=3;count=0; 
	}
}

 

void main()
{	
	motinit();
	infrayinit();
	lcd_init();
	wave_init();
	servoinit();
	while(1)
	{ 
		infray();
		wave();
		servo();
		spreed();
		 direction(dirdata);	
	}
}


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值