❶ 单片机电控板 +红外摇按的程序
说两点,第一,周期问题,不管是二进制的1还是二进制的0,他们的周期都是20us,而不是你说的0是10us.
第二,你要学会自己分析一下电路和波形,以20us为周期,划分四个时段来看,0和1的区别在于
第0.25-0.5,0.5-0.75两个时段内.
而你是在外中断引脚上接的,这更加利于检测,所以,在检测到下降沿之后,延时一段时间,再查看引脚的状态,如果是低电平,表示当前是0,如果是高电平,表示当前是1,具体延时多长时间呢,要取0.5--0.75之间的一个数字,为什么,你可以自己画一下波形,然后算一下,就很容易了,这里,我们取严格的0.5个周期,也就是说,10us的延时,这个延时要非常精确,必要时,甚至要关掉总中断,延时太长,不利于实时性,延时太短,有可能误判断.
下面是我的程序:
void Delay_10us(void) 必须严格控制在 10-11us之间
sbit EIX = P3^0;
unsigned short Ctrl_Char =0;//控制字
unsigned char Cmd = 0;
void ISR_EXT0(void) interrupt 0 //----根据情况选择
{
EX0 = 0; //---关中断
Ctrl_Char <<= 1; //---把最后一位让出来
Delay_10us();
EX0 = 1; //---开中断
if(EIX) Ctrl_Char ++;
}
在中断以外这样写:
以下函数要保证实时执行,以免错过一次命令的读取
void Juge_CtrChar(void)
{
if((Ctrl_Char &0xFF00) == 0xA500)
{
Cmd = Ctrl_Char; //----这里是利用隐性转换
Ctrl_Char =0; //---读取成功,则清零本次的解码
}
}
❷ 求一个单片机红外遥控器控制设计的源程序
程序太长,须仔细研究。
#definemain_GLOBALS
#include<reg51.h>
#include"main.h"
#include"SAA3010.h"
unsignedcharcounter;
unsignedchartemp;
sbitP2_0=0xA0;
sbitP2_1=0xA1;
sbitP2_2=0xA2;
sbitP2_3=0xA3;
voiddecode_init(void)
{
load_code_detected=0;
repeat_code_detected=0;
decode_error=0;
temp1=0;
temp2=0;
temp3=0;
temp4=0;
TH1=0;
TL1=0;
TR1=0;
}
voidmain(void)
{
EX0=1;
IT0=1;
TMOD=0x11;
ET0=1;
TH0=128;
TL0=0;
TR0=1;
P0=0;
TH1=0;
TL1=0;
decode_init();
EA=1;
counter=0;
data_available=0;
while(1)
{
if(data_available==1)
{
if(key_code==01)//display"1"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0xf9;
}
if(key_code==0x02)//display"2"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0xa4;
}
if(key_code==0x03)//display"3"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0xb0;
}
if(key_code==0x04)//display"4"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0x99;
}
if(key_code==0x05)//display"5"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0x92;
}
if(key_code==0x06)//display"6"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0x82;
}
if(key_code==0x07)//display"7"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0xf8;
}
if(key_code==0x08)//display"8"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0x80;
}
if(key_code==0x09)//display"9"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0x90;
}
if(key_code==0x00)//display"0"
{
P2_0=0;//选中四个数码管
P2_1=0;
P2_2=0;
P2_3=0;
P0=0xc0;
}
data_available=0;
}
}
}
//**********************************************************************************************************************
voidkey_isr(void)interrupt0
{
EX0=0;//立即关闭外部中断,转为查询方式解码
led=0;//开LED表示收到红外信号
temp=SAA3010_decode();//解码
if(temp==1)gotokey_isr_exit;
counter=0;
key_isr_exit:
decode_init();
led=1;
EX0=1;
}
voidTimer0_isr(void)interrupt1
{
counter++;
if(counter>3)
{
if(led==0)led=1;
counter=0;
}
ET0=1;
}
#defineSAA3010_GLOBALS
#include<reg51.h>
#include"SAA3010.h"
#include"main.h"
//===========================================================================================================
//该函数的作用是每调用一次就在temp1-4组成的32bit长度的最低位上移入
//一个0或者1,数据由bitdata确定
voidSAA3010_cycle_data(unsignedcharbitdata)
{
temp4=temp4<<1;
if((temp3&0x80)==1)temp4=temp4|0x01;
elsetemp4=temp4&0xfe;
temp3=temp3<<1;
if((temp2&0x80)==1)temp3=temp3|0x01;
elsetemp3=temp3&0xfe;
temp2=temp2<<1;
if((temp1&0x80)==1)temp2=temp2|0x01;
elsetemp2=temp2&0xfe;
temp1=temp1<<1;
if(bitdata==1)temp1=temp1|0x01;
elsetemp1=temp1&0xfe;
}
//===========================================================================================================
//解码出错返回1,对则返回0
unsignedcharSAA3010_decode(void)
{
//-----------------------------------------------------------------------------------------------------------
unsignedcharcount=0;
TR1=1;//启动计时
while(1)
{
while(ir_receive==0);//等待电平变高,不需要超时监测
TR1=0;//高电平(对发射电路而言)测试结束
high_level_time=TH1*256+TL1;//记录高电平的数据
//-----------------------------------------------------------------------------------------------------------
TH1=0; TL1=0;TR1=1;//启动对低电平的测试
//-----------------------------------------------------------------------------------------------------------
//处理低电平
if((high_level_time<750)||(high_level_time>1800))return1;//不是合格的电平
if((high_level_time>750)&&(high_level_time<1000)){SAA3010_cycle_data(0);count+=1;}//移入一个0
if((high_level_time>1500)&&(high_level_time<1800)){SAA3010_cycle_data(0);SAA3010_cycle_data(0);count+=2;}//移入两个0
while(ir_receive==1)//等待电平变低
{
if(TH1>0x08)break;//高电平超时,正常情况下是测试结束,异常时则是出错
}
TR1=0;//低电平(对发射电路而言)测试结束
if(TH1>0x08){break;}
low_level_time=TH1*256+TL1;//保存低电平的数据
TH1=0; TL1=0; TR1=1; //为增加计时的准确性,数据的处理都是在计时过程里
//-----------------------------------------------------------------------------------------------------------
//处理高电平
if((low_level_time<750)||(low_level_time>1800))return1;//不是合格的电平
if((low_level_time>750)&&(low_level_time<1000)){SAA3010_cycle_data(1);count+=1;}//移入一个0
if((low_level_time>1500)&&(low_level_time<1800)){SAA3010_cycle_data(1);SAA3010_cycle_data(1);count+=2;}//移入两个0
}
if(count==26){SAA3010_cycle_data(1);count++;}
if(count!=27)return1;
led=0;
//提取按健信息
key_code=0;
if((temp1>>1)&0x01)key_code=key_code|0x01;
elsekey_code=key_code&0xfe;
if((temp1>>3)&0x01)key_code=key_code|0x02;
elsekey_code=key_code&0xfd;
if((temp1>>5)&0x01)key_code=key_code|0x04;
elsekey_code=key_code&0xfb;
if((temp1>>7)&0x01)key_code=key_code|0x08;
elsekey_code=key_code&0xf7;
if((temp2>>1)&0x01)key_code=key_code|0x10;
elsekey_code=key_code&0xef;
if((temp2>>3)&0x01)key_code=key_code|0x20;
elsekey_code=key_code&0xdf;
//提取系统信息
sys_code=0;
if((temp2>>5)&0x01)sys_code=sys_code|0x01;
elsesys_code=sys_code&0xfe;
if((temp2>>7)&0x01)sys_code=sys_code|0x02;
elsesys_code=sys_code&0xfd;
if((temp3>>1)&0x01)sys_code=sys_code|0x04;
elsesys_code=sys_code&0xfb;
if((temp3>>3)&0x01)sys_code=sys_code|0x08;
elsesys_code=sys_code&0xf7;
if((temp3>>5)&0x01)sys_code=sys_code|0x10;
elsesys_code=sys_code&0xef;
if((temp3>>7)&0x01)sys_code=sys_code|0x20;
elsesys_code=sys_code&0xdf;
data_available=1;
return0;
}
//===========================================================================================================
❸ 51单片机 红外控制程序中 if (IRCOM[2]!=~IRCOM[3]) { EX0=1; return; }是什么意思
应该是接到的数据,正反码吧,因为红外线可能受干扰,需要校验 一般会采取发两个字节,这两个字节的值是反码关系,接受如果发现不是就认为数据出错。。。。。EX0的作用就不知道了,也许是某个IO,猜测而已,你如果把源码发出来可能分析一下,凭空写个这东西很难知道干啥的。。。
❹ 单片机红外控制程序,上电后数码管显示8个0,遥控按每反应,这是什么原因啊,求高手赐教,一定给你加分
#include <reg51.h>
sbit IRIN = P3^2; //红外接收器数据线
sbit RELAY= P1^4; //继电器驱动线
sbit BEEP = P1^5; //蜂鸣器驱动线
void IRdelay(char x); //x*0.14MS
void beep();
unsigned char IRCOM[7];
extern unsigned char Y0;
/*******************************************************************/
void IRInit()
{
IE |= 0x81; //允许总中断中断,使能 INT0 外部中断
TCON |= 0x01; //触发方式为脉冲负边沿触发
IRIN=1; //I/O口初始化
RELAY=1;
}
/**********************************************************/
void IR_IN(void) interrupt 0
{
unsigned char j,k,N=0;
EX0 = 0;
IRdelay(15);
if (IRIN==1)
{ EX0 =1;
return;
}
//确认IR信号出现
while (!IRIN) //等IR变为高电平,跳过9ms的前导低电平信号。
{IRdelay(1);}
for (j=0;j<4;j++) //收集四组数据
{
for (k=0;k<8;k++) //每组数据有8位
{
while (IRIN) //等 IR 变为低电平,跳过4.5ms的前导高电平信号。
{IRdelay(1);}
while (!IRIN) //等 IR 变为高电平
{IRdelay(1);}
while (IRIN) //计算IR高电平时长
{
IRdelay(1);
N++;
if (N>=30)
{ EX0=1;
return;} //0.14ms计数过长自动离开。
} //高电平计数完毕
IRCOM[j]=IRCOM[j] >> 1; //数据最高位补“0”
if (N>=8) {IRCOM[j] = IRCOM[j] | 0x80;} //数据最高位补“1”
N=0;
}//end for k
}//end for j
if (IRCOM[2]!=~IRCOM[3])
{ EX0=1;
return; }
IRCOM[5]=IRCOM[2] & 0x0F; //取键码的低四位
IRCOM[6]=IRCOM[2] >> 4; //右移4次,高四位变为低四位
if(IRCOM[5]>9)
{ IRCOM[5]=IRCOM[5]+0x37;}
else
IRCOM[5]=IRCOM[5]+0x30;
if(IRCOM[6]>9)
{ IRCOM[6]=IRCOM[6]+0x37;}
else
IRCOM[6]=IRCOM[6]+0x30;
if(IRCOM[2]==0x4c) RELAY=0; //打开继电器
else RELAY=1; //关闭继电器
beep();
/*
;================================
;****** 红外遥控器键值表 ******
; 4c 0e 4d 0d
; 0f 4f 4e 0c
; 0b 4b 4a 08
; 48 12 16 4C
; 40 48 04 00
; 02 05 54 4D
; 0A 1E 0E 1A
; 1C 14 0F 0C
;================================ */
Y0=0;
switch(IRCOM[2])
{
case 0x4f: Y0=1; break;
case 0x4e: Y0=2; break;
case 0x0c: Y0=3; break;
case 0x4b: Y0=4; break;
case 0x4a: Y0=5; break;
case 0x08: Y0=6; break;
case 0x0a: Y0=7; break;
case 0x49: Y0=8; break;
case 0x09: Y0=9; break;
case 0x48: Y0=0; break;
}
EX0 = 1;
}
/**********************************************************/
void IRdelay(unsigned char x) //x*0.14MS
{
unsigned char i;
while(x--)
{
for (i = 0; i<13; i++) {}
}
}
void beep()
{
unsigned char i;
for (i=0;i<100;i++)
{
IRdelay(4);
BEEP=!BEEP; //BEEP取反
}
BEEP=1; //关闭蜂鸣器
}
❺ 51单片机红外线遥控LED程序,疑惑中!!!!!
红外收发中,IRDATA[2]与IRDATA[3]是取反的关系。也就是说两个数对应各位前者为1后者就为0
其余的问题,都是根据红外接收时序来编的程序,以下总结以下红外收发时序供参考,你读懂就可以理解了。
采用脉宽调制的串行码,以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示二进制的"0";;以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示二进制的"1“上述"0"和"1"组成的32位二进制码经38kHz的载频进行二次调制以提高发射效率,达到降低电源功耗的目的。然后再通过红外发射二极管产生红外线向空间发射遥控编码是连续的32位二进制码组,其中前16位为用户识别码,能区别不同的电器设备,防止不同机种遥控码互相干扰。该芯片的用户识别码固定为十六进制01H后16位为8位操作码(功能码)及其反码。
发射代码由一个起始码(9ms),一个结果码(4.5ms),低8位地址码(9ms~18ms),高8位地址码(9ms~18ms),8位数据码(9ms~18ms)和这8位数据的反码(9ms~18ms)组成。
解码的关键是如何识别"0"和"1",接收端而言,"0"是0.56ms的高+0.56ms的低。"1"是1.68ms的高+0.56ms的低。所以可以根据高电平的宽度区别"0"和"1"。当高电平出现时开始延时,0.56ms以后,若读到的电平为低,说明该位为"0",反之则为"1",为了可靠起见,延时必须比0.56ms长些,但又不能超过1.12ms,否则如果该位为"0",读到的已是下一位的高电平,因此取(1.12ms+0.56ms)/2=0.84ms最为可靠,一般取0.84ms左右均可。为了共用引导部分延时程序,一般用0.9ms延时。
由此可见,有效数据是4字节(32位)。前两个字节可定义用户编码,后两个字节分别是真正的数据及其反码。
❻ 单片机控制红外程序
给你一个PIC写的接收红外遥控器的程序。
//******************************************************
//mcu: PIC16F883
//author:
//data:
//ver: 1.0
//********************************************************
#include <pic.h>
#include <pic16f887.h>
#include "main.h"
// 函数声明部份
void interrupt IRQ_TEST(void);
void fun_intset(void);
void set_port(void);
void fun_timerset(void);
void delay_10us(void);
void delay_130us(void);
void test_remote(void);
// 主函数
void main()
{
set_port();
fun_timerset();
fun_intset();
delay_10ms();
ram_ini();
GIE=1;
while(1)
{
asm("clrwd");
test_remote();
}
}
//端口I/O定义子程序
void set_port()
{
ANSEL =0X00; //porta io port
ANSELH=0X00; //portb io port
//TRISX BIT=1, PORTX AS INPUT
TRISA = 0x0F; //RA0-RA3 INPUT ,RA4-RA5 OUTPUT
TRISB = 0x07; //RB0 PASS0, RB1-RB2 INPUT ,RB3-RB6 OUTPUT
TRISC = 0x0F;
// TRISC = 0x03; //RC0-RC1 INPUT,RC5-RC6 OUTPUT , RC3 RC4 EEPROM SCL SDA
PORTC=0;
PORTA=0XFF;
// WPUB=0XFF;
PORTB=0X07;
}
//中断子程序
void interrupt IRQ_TEST(void)
{
if(T0IF) //Timer0中断服务子程序
{
TMR0=0xce;
T0IF=0;
timer_100us++; //100us
}
if(INTF==1)
{
INTF=0;
timer_rec=timer_100us;
timer_100us=0;
flag_rec_remote=1;
}
}
//定时器0初始化设置子程序
void fun_timerset()
{
//Timer0初始化设置
PSA=0;//Timer0 使用预分频器
//Timer0选择分频率为1:2
PS0=0;PS1=0;PS2=0;
//内部时钟定时方式,定时时间:200uS,误差:0uS
T0CS=0;
// TMR0=0x9b;
TMR0=0xce; //100us
}
/*
//定时器2初始化设置子程序
void fun_timer2set()
{
//Timer2初始化设置
//timer2使用预分频率1:1
T2CKPS0=0;T2CKPS1=0;
//timer2使用后分频率1:1
TOUTPS0=0;
TOUTPS1=0;
TOUTPS2=0;
TOUTPS3=0;
//定时器2定时时间为:200uS,误差:0uS
TMR2=0x38;
TMR2ON=1;
}
*/
//中断允许设置子程序
void fun_intset()
{
T0IE=1;//Tiemr0中断允许
INTEDG=0; //RB0 FALLAGE INTERRUPT
INTE=1;
INTF=0;
}
void ram_ini(void)
{
}
//------------------------------------------------------------
//*****************************************************************************
void delay_10us(void)
{
NOP;
NOP;
NOP;
NOP;
NOP;
NOP;
}
void delay_130us(void)
{
uchar i;
for(i=0;i<13;i++)
{
NOP;
NOP;
NOP;
}
}
void delay_10ms(void)
{
unsigned int i;
for(i=0;i<1000;i++)
{
NOP;
NOP;
NOP;
NOP;
NOP;
}
}
void test_remote(void)
{
uchar rec_ok,i;
if(flag_rec_remote==1)
{
flag_rec_remote=0;
// remote_buf[rec_byte]=timer_rec; //test use
// rec_byte++; //test use
// return; //test use
if(flag_rec_head==0)
{
if(timer_rec>=90&&timer_rec<=140) //135
{
flag_rec_head=1;
rec_byte=0;
rec_bit=0;
rec_buf=0;
return;
}
}
else
{
if(timer_rec>=8&&timer_rec<=12) //11.2
{
rec_buf=(rec_buf>>1);
rec_buf=rec_buf&0x7f;
}
else if(timer_rec>=18&&timer_rec<=22) //22
{
rec_buf=(rec_buf>>1);
rec_buf=rec_buf|0x80;
}
else if(timer_rec>=90&&timer_rec<=140)
{
flag_rec_head=1;
rec_byte=0;
rec_bit=0;
rec_buf=0;
return;
}
else
{
flag_rec_head=0;
rec_byte=0;
rec_bit=0;
rec_buf=0;
return;
}
}
if(flag_rec_head==1)
{
rec_bit++;
if(rec_bit==8)
{
remote_buf[rec_byte]=rec_buf;
rec_byte++;
rec_bit=0;
if(rec_byte==4)
{
NOP;
NOP;
NOP;
NOP;
rec_byte=0;
rec_bit=0;
rec_buf=0;
flag_rec_head=0;
}
}
}
}
}
❼ 基于51单片机红外遥控代码(C语言)
以下文件是51单片机实现遥控解码,通过数码管显示键码的程序,P0口驱动数码管段选,p2.6和p2.7为数码管位选,接收头连到P3.2口。此程序以通过验证,可以直接编译使用,另外还有一个继电器和蜂鸣器的控制,不用可以屏蔽掉。
;********************************************************************************
;* 描述: *
;* 遥控键值读取器 *
;* 数码管显示, P0口为数码管的数据口 *
;* *
;********************************************************************************
;遥控键值解码-数码管显示 *
;********************************************************************************/
#include <reg51.h>
#include <intrins.h>
void IR_SHOW();
void delay(unsigned char x);//x*0.14MS
void delay1(unsigned char ms);
void beep();
sbit IRIN = P3^2;
sbit BEEP = P3^7;
sbit RELAY= P1^3;
sbit GEWEI= P2^7;
sbit SHIWEI= P2^6;
unsigned char IRCOM[8];
unsigned char code table[16] =
{0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e};
main()
{
IE = 0x81; //允许总中断中断,使能 INT0 外部中断
TCON = 0x1; //触发方式为脉冲负边沿触发
delay(1);
IRIN=1;
BEEP=1;
RELAY=1;
for(;;)
{
IR_SHOW();
}
} //end main
void IR_IN() interrupt 0 using 0
{
unsigned char i,j,k,N=0;
EA = 0;
I1:
for (i=0;i<4;i++)
{
if (IRIN==0) break;
if (i==3) {EA =1;return;}
}
delay(20);
if (IRIN==1) goto I1; //确认IR信号出现
while (!IRIN) //等 IR 变为高电平
{delay(1);}
for (j=0;j<4;j++)
{
for (k=0;k<8;k++)
{
while (IRIN) //等 IR 变为低电平
{delay(1);}
while (!IRIN) //等 IR 变为高电平
{delay(1);}
while (IRIN) //计算IR高电平时长
{
delay(1);
N++;
if (N>=30) {EA=1;return;}
}
IRCOM[j]=IRCOM[j] >> 1;
if (N>=8) {IRCOM[j] = IRCOM[j] | 0x80;}
N=0;
}//end for k
}//end for j
if (IRCOM[2]!=~IRCOM[3]) {EA=1;return;}
IRCOM[5]=IRCOM[2] & 0x0F;
IRCOM[6]=IRCOM[2] & 0xF0;
IRCOM[6]=IRCOM[6] >> 4;
beep();
EA = 1;
}
void IR_SHOW()
{
P0 = table[IRCOM[5]];
GEWEI = 0;
SHIWEI = 1;
delay1(4);
P0 = table[IRCOM[6]];
SHIWEI = 0;
GEWEI = 1;
delay1(4);
}
void beep()
{
unsigned char i;
for (i=0;i<100;i++)
{
delay(5);
BEEP=!BEEP;
}
BEEP=1;
}
void delay(unsigned char x)//x*0.14MS
{
unsigned char i;
while(x--)
{
for (i = 0; i<13; i++) {}
}
}
void delay1(unsigned char ms)
{
unsigned char i;
while(ms--)
{
for(i = 0; i<120; i++)
{
_nop_();
_nop_();
_nop_();
_nop_();
}
}
}
❽ 51单片机红外遥控程序是什么
#include
❾ 求51单片机红外遥控程序
sbitIR_OUT=P3^2;
unsignedcharSigInfo[4];//存储红外按键编码,SigInfo[2]为按键值
voidCmd_Require(void);
voidINT_Ext0()interrupt1//外部中断0
{
u8i,j;
u16time=8000;
EA=0;
for(i=0;i<10;i++){
delay0_7ms();
if(IR_OUT){
EA=1;
return;
}
}
while(!IR_OUT);
delay2_5ms();
if(!IR_OUT){
EA=1;
return;
}
while(IR_OUT&&time){
time--;
}
time=8000;
for(i=0;i<4;i++){
for(j=0;j<8;j++){
SigInfo[i]>>=1;
while(!IR_OUT);
delay0_7ms();
if(IR_OUT){
SigInfo[i]|=0x80;
while(IR_OUT&&time){
time--;
}
time=8000;
}
}
}
Cmd_Require();
EA=1;
}
voidCmd_Require(void)
{
switch(SigInfo[2]){
case0x0c:
//此处代码随意
break;
case0x18:
//...
break;
//...
default:
break;
}
}
延时没写,因单片机而异的,要精确点才能解码。
❿ 51单片机的红外遥控小车设计和制作的C语言程序
论文摘要:本文介绍一款红外线遥控小车,以AT89S51单片机为核心控制器,用L289驱动直流电机工作,控制小车的运行。本款小车具有红外线遥控手动驾驶、自动驾驶、寻迹前进等功能。本系统采用模块化设计,软件用C语言编写。转贴于 51论 文网 www.51lun-wen.cn一、设计任务和要求以AT98C51单片机为核心,制作一款红外遥控小车,小车具有自动驾驶,手动驾驶和循迹前进等功能。自动驾驶时,前进过程中可以避障。手动驾驶时,遥控控制小车前进、后退、左转、右转、加速等操作。寻迹前进时小车还可以按照预先设计好的轨迹前进。转贴于 51论 文网 www.51lun-wen.cn二、系统组成及工作原理本系统由硬件和软件两部分组成。硬件部分主要完成红外编码信号的发射和接受、障碍物检测、轨迹检测、直流电机运行的发生等功能。软件主要完成信号的检测和处理、设备的驱动及控制等功能。AT89S51单片机查询红外信号并解码,查询各个检测部分输入的信号,并进行相应处理,包括电机的正反转,判断是否遇到障碍物,判断是否小车其那金中有出轨等。系统结构框图如图1所示。转贴于 51论 文网 www.51lun-wen.cn图1 系统结构框图三、主要硬件电路1、遥控发射器电路该电路的主要控制器件为遥控器芯片HT6221,如图2所示。HT6221将红外码调制成38KHZ的脉冲信号通过红外发射二极管发出红外编码。图2中D1是红外发射二极管,D2是按键指示灯,当有按键按下时D2点亮。