❶ 51單片機紅外遙控程序
#include<reg52.h>
#include<intrins.h>
#define uchar unsigned char
#define uint unsigned int
sbit lcden = P2^7;
sbit lcdrs = P2^6;
sbit lcdwr = P2^5;
sbit IR = P3^2;
uchar IRCOM[6];//數組,用於存儲紅外編碼
uchar code table1[] = "remote control";
uchar code table2[] = "CODE:";
void delayms(uchar x)// 延時x*0.14ms
{
uchar i;
while(x--)
for(i=0;i<13;i++){}
}
void delay(uchar x) //延時xms
{
uchar i,j;
for(i=x;i>0;i--)
for(j=110;j>0;j--);
}
/****************************LCD部分***********************************************/
void write_com(uchar com)
{
lcden = 0;
lcdrs = 0;
lcdwr = 0;
P0 = com;
delay(5);
lcden = 1;
delay(5);
lcden = 0; //別忘了lcden拉低
}
void write_date(uchar date)
{
lcden = 0;
lcdrs = 1;
lcdwr = 0;
P0 = date;
delay(5);
lcden = 1;
delay(5);
lcden = 0;
}
void lcd_init(void)
{
lcden = 0;
lcdrs = 0;
lcdwr = 0;
delay(5);
write_com(0x38);
write_com(0x0c);
write_com(0x06);
write_com(0x01);
}
/*****************main()************************/
void main(void)
{
uchar count=0;
IR = 1;
lcd_init();
write_com(0x80);
while(table1[count]!='\0')
{
write_date(table1[count]);
count++;
delay(5);
}
count = 0;
write_com(0x80+0x40);
while(table2[count]!='\0')
{
write_date(table2[count]);
count++;
delay(5);
}
IE = 0x81; //開中斷
TCON = 0x01;//脈沖負邊沿觸發
while(1);
}
/*********************紅外中斷**************************/
void IR_time() interrupt 0
{
uchar i,j,TimeNum=0;//TimeNum用來計IR高電平次數 從而判斷是0還是1
EX0 = 0; //關閉中斷
delayms(5);
if(1 == IR)
{
EX0 = 1;
return;
}
while(!IR) //跳過9ms前導低電平
delayms(1);
for(i=0;i<4;i++)
{
for(j=0;j<8;j++)
{
while(IR) //跳過4.5ms的前導高電平
delayms(1);
while(!IR) //跳過0.56ms的低電平
delayms(1);
while(IR)
{
TimeNum++; //計時高電平時間從而判斷讀取的是0還是1
delayms(1);
}
if(TimeNum>=30)//按鍵按下時間過長 跳過
{
EX0 = 1;
return;
}
IRCOM[i] = IRCOM[i]>>1;
if(TimeNum >= 8) //8*0.14ms 這時讀取的是1;
{
IRCOM[i] = IRCOM[i]|0x80;
}
TimeNum = 0;
}
}
if(IRCOM[2]!=~IRCOM[3])//判斷八位數據和八位數據反碼是否相等
{
EX0 = 1;
return;
}
IRCOM[4] = IRCOM[2]&0x0f;//取低四位
IRCOM[5] = IRCOM[2]>>4; //IRCOM[5]取IRCOM[2]高四位
if(IRCOM[4] > 9) //轉換成字元
{
IRCOM[4] = IRCOM[4] + 0x37;
}
else
IRCOM[4] = IRCOM[4] + 0x30;
if(IRCOM[5] > 9)
{
IRCOM[5] = IRCOM[5] + 0x37;
}
else
IRCOM[5] = IRCOM[5] + 0x30;
delay(5);
write_com(0x80 + 0x40 + 5);
write_date(IRCOM[5]);
write_date(IRCOM[4]);
EX0 = 1; //重新開啟外部中斷
}
❷ 基於單片機的紅外萬能空調遙控的源代碼
我也認為這個該付費的,網上上搜了很多,找不到合適的,自己研究了很長時間,不很成功,不過已能學習幾個按鍵,方法就是用定時器把所有的高低電平持續時間測出來,存入EEPROM中,並不識別什麼是0,什麼是1,原樣保存,原樣使用,這樣對單片機的速度和內存要求高一些,空調器的控制碼有100多位,實際上能學習一個按鍵,就成功了一半,但下一部還是要與析它的代碼規律,以舍棄一些不必要的代碼,空調器常用的按鍵就沒幾個,但不同工作條件下同一個按鍵發出的控制碼是不同的,全部存儲,不僅浪費存儲空間,而且會給查找調用帶來不方便
❸ 基於單片機的紅外遙控收發系統的設計與實現
紅外遙控一般採用38KHz載波
❹ 51單片機最小系統怎樣連接紅外遙控模塊
紅外信號的發射由紅外發射電路中的紅外發光二極體完成,通常情況下為了提高抗干擾能力與降低電源消耗,遙控器將遙控信號(二進制脈沖碼)調制在載波(載波是傳送信息的物理基礎和承載工具)上經放大後發送至紅外二極體,再由二極體轉換為紅外信號發送出去。遙控器上不同的按鍵有著不一樣的鍵值,按下相對應的鍵,紅外二極體就會發送對應的信號,接收裝置接收到信號後會對信號進行信號解調後會得到相應按鍵的鍵值,再根據不同的鍵值執行相應的操作。
❺ 基於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_();
}
}
}
❻ 課程設計:紅外遙控解碼器(基於單片機AT89C51),能同過數碼管把相應的按鍵顯示出來~ 各位高手幫幫忙
//實例97:用紅外遙控器控制繼電器
#include<reg51.h> //包含單片機寄存器的頭文件
sbit IR=P3^2; //將IR位定義為P3.2引腳
unsigned char a[4]; //儲存用戶碼、用戶反碼與鍵數據碼、鍵數據反碼
unsigned int LowTime,HighTime; //儲存高、低電平的寬度
sbit Relay=P1^3; //將Relay位定義為P1.3引腳
/************************************************************
函數功能:對4個位元組的用戶碼和鍵數據碼進行解碼
說明:解碼正確,返回1,否則返回0
出口參數:dat
*************************************************************/
bit DeCode(void)
{
unsigned char i,j;
unsigned char temp; //儲存解碼出的數據
for(i=0;i<4;i++) //連續讀取4個用戶碼和鍵數據碼
{
for(j=0;j<8;j++) //每個碼有8位數字
{
temp=temp>>1; //temp中的各數據位右移一位,因為先讀出的是高位數據
TH0=0; //定時器清0
TL0=0; //定時器清0
TR0=1; //開啟定時器T0
while(IR==0) //如果是低電平就等待
; //低電平計時
TR0=0; //關閉定時器T0
LowTime=TH0*256+TL0; //保存低電平寬度
TH0=0; //定時器清0
TL0=0; //定時器清0
TR0=1; //開啟定時器T0
while(IR==1) //如果是高電平就等待
;
TR0=0; //關閉定時器T0
HighTime=TH0*256+TL0; //保存高電平寬度
if((LowTime<370)||(LowTime>640))
return 0; //如果低電平長度不在合理范圍,則認為出錯,停止解碼
if((HighTime>420)&&(HighTime<620)) //如果高電平時間在560微秒左右,即計數560/1.085=516次
temp=temp&0x7f; //(520-100=420, 520+100=620),則該位是0
if((HighTime>1300)&&(HighTime<1800)) //如果高電平時間在1680微秒左右,即計數1680/1.085=1548次
temp=temp|0x80; //(1550-250=1300,1550+250=1800),則該位是1
}
a[i]=temp; //將解碼出的位元組值儲存在a[i]
}
if(a[2]=~a[3]) //驗證鍵數據碼和其反碼是否相等,一般情況下不必驗證用戶碼
return 1; //解碼正確,返回1
}
/************************************************************
函數功能:執行遙控功能
*************************************************************/
void Function(void)
{
Relay=!Relay; //對P1.3引腳取反,控制繼電器的吸合、釋放
}
/************************************************************
函數功能:主函數
*************************************************************/
void main()
{
EA=1; //開啟總中斷
EX0=1; //開外中斷0
ET0=1; //定時器T0中斷允許
IT0=1; //外中斷的下降沿觸發
TMOD=0x01; //使用定時器T0的模式1
TR0=0; //定時器T0關閉
while(1) //等待紅外信號產生的中斷
;
}
/************************************************************
函數功能:紅外線觸發的外中斷處理函數
*************************************************************/
void Int0(void) interrupt 0 using 0
{
EX0=0; //關閉外中斷0,不再接收二次紅外信號的中斷,只解碼當前紅外信號
TH0=0; //定時器T0的高8位清0
TL0=0; //定時器T0的低8位清0
TR0=1; //開啟定時器T0
while(IR==0) //如果是低電平就等待,給引導碼低電平計時
;
TR0=0; //關閉定時器T0
LowTime=TH0*256+TL0; //保存低電平時間
TH0=0; //定時器T0的高8位清0
TL0=0; //定時器T0的低8位清0
TR0=1; //開啟定時器T0
while(IR==1) //如果是高電平就等待,給引導碼高電平計時
;
TR0=0; //關閉定時器T0
HighTime=TH0*256+TL0; //保存引導碼的高電平長度
if((LowTime>7800)&&(LowTime<8800)&&(HighTime>3600)&&(HighTime<4700))
{
//如果是引導碼,就開始解碼,否則放棄,引導碼的低電平計時
//次數=9000us/1.085=8294, 判斷區間:8300-500=7800,8300+500=8800.
if(DeCode()==1)
Function(); //如果滿足條件,執行遙控功能
}
EX0=1; //開啟外中斷EX0
}
❼ 用51單片機製作學習型紅外遙控器的原理
以下是程序,調試成功,LCD1602顯示
//本解碼程序適用於NEC的upd6121及其兼容晶元的解碼,支持大多數遙控器實驗板採用11.0592MHZ晶振
#include<reg52.h>//包含單片機寄存器的頭文件
#include<intrins.h>//包含_nop_()函數定義的頭文件
sbitIR=P3^2;//將IR位定義為P3.2引腳
sbitRS=P2^0;//寄存器選擇位,將RS位定義為P2.0引腳
sbitRW=P2^1;//讀寫選擇位,將RW位定義為P2.1引腳
sbitE=P2^2;//使能信號位,將E位定義為P2.2引腳
sbitBF=P0^7;//忙碌標志位,,將BF位定義為P0.7引腳
sbitBEEP=P3^6;//蜂鳴器控制埠P36
unsignedcharflag;
unsignedcharcodestring[]={"1602IR-CODETEST"};
unsignedchara[4];//儲存用戶碼、用戶反碼與鍵數據碼、鍵數據反碼
unsignedintLowTime,HighTime;//儲存高、低電平的寬度
/*****************************************************
函數功能:延時1ms
***************************************************/
voiddelay1ms()
{
unsignedchari,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++)
;
}
/*****************************************************
函數功能:延時若干毫秒
入口參數:n
***************************************************/
voiddelay(unsignedcharn)
{
unsignedchari;
for(i=0;i<n;i++)
delay1ms();
}
/*********************************************************/
voidbeep()//蜂鳴器響一聲函數
{
unsignedchari;
for(i=0;i<100;i++)
{
delay1ms();
BEEP=!BEEP;//BEEP取反
}
BEEP=1;//關閉蜂鳴器
delay(250);//延時
}
/*****************************************************
函數功能:判斷液晶模塊的忙碌狀態
返回值:result。result=1,忙碌;result=0,不忙
***************************************************/
unsignedcharBusyTest(void)
{
bitresult;
RS=0;//根據規定,RS為低電平,RW為高電平時,可以讀狀態
RW=1;
E=1;//E=1,才允許讀寫
_nop_();//空操作
_nop_();
_nop_();
_nop_();//空操作四個機器周期,給硬體反應時間
result=BF;//將忙碌標志電平賦給result
E=0;
returnresult;
}
/*****************************************************
函數功能:將模式設置指令或顯示地址寫入液晶模塊
入口參數:dictate
***************************************************/
voidWriteInstruction(unsignedchardictate)
{
while(BusyTest()==1);//如果忙就等待
RS=0;//根據規定,RS和R/W同時為低電平時,可以寫入指令
RW=0;
E=0;//E置低電平(根據表8-6,寫指令時,E為高脈沖,
//就是讓E從0到1發生正跳變,所以應先置"0"
_nop_();
_nop_();//空操作兩個機器周期,給硬體反應時間
P0=dictate;//將數據送入P0口,即寫入指令或地址
_nop_();
_nop_();
_nop_();
_nop_();//空操作四個機器周期,給硬體反應時間
E=1;//E置高電平
_nop_();
_nop_();
_nop_();
_nop_();//空操作四個機器周期,給硬體反應時間
E=0;//當E由高電平跳變成低電平時,液晶模塊開始執行命令
}
/*****************************************************
函數功能:指定字元顯示的實際地址
入口參數:x
***************************************************/
voidWriteAddress(unsignedcharx)
{
WriteInstruction(x|0x80);//顯示位置的確定方法規定為"80H+地址碼x"
}
/*****************************************************
函數功能:將數據(字元的標准ASCII碼)寫入液晶模塊
入口參數:y(為字元常量)
***************************************************/
voidWriteData(unsignedchary)
{
while(BusyTest()==1);
RS=1;//RS為高電平,RW為低電平時,可以寫入數據
RW=0;
E=0;//E置低電平(根據表8-6,寫指令時,E為高脈沖,
//就是讓E從0到1發生正跳變,所以應先置"0"
P0=y;//將數據送入P0口,即將數據寫入液晶模塊
_nop_();
_nop_();
_nop_();
_nop_();//空操作四個機器周期,給硬體反應時間
E=1;//E置高電平
_nop_();
_nop_();
_nop_();
_nop_();//空操作四個機器周期,給硬體反應時間
E=0;//當E由高電平跳變成低電平時,液晶模塊開始執行命令
}
/*****************************************************
函數功能:對LCD的顯示模式進行初始化設置
***************************************************/
voidLcdInitiate(void)
{
delay(15);//延時15ms,首次寫指令時應給LCD一段較長的反應時間
WriteInstruction(0x38);//顯示模式設置:16×2顯示,5×7點陣,8位數據介面
delay(5);//延時5ms
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x38);
delay(5);
WriteInstruction(0x0C);//顯示模式設置:顯示開,有游標,游標閃爍
delay(5);
WriteInstruction(0x06);//顯示模式設置:游標右移,字元不移
delay(5);
WriteInstruction(0x01);//清屏幕指令,將以前的顯示內容清除
delay(5);
}
/************************************************************
函數功能:對4個位元組的用戶碼和鍵數據碼進行解碼
說明:解碼正確,返回1,否則返回0
出口參數:dat
*************************************************************/
bitDeCode(void)
{
unsignedchari,j;
unsignedchartemp;//儲存解碼出的數據
for(i=0;i<4;i++)//連續讀取4個用戶碼和鍵數據碼
{
for(j=0;j<8;j++)//每個碼有8位數字
{
temp=temp>>1;//temp中的各數據位右移一位,因為先讀出的是高位數據
TH0=0;//定時器清0
TL0=0;//定時器清0
TR0=1;//開啟定時器T0
while(IR==0)//如果是低電平就等待
;//低電平計時
TR0=0;//關閉定時器T0
LowTime=TH0*256+TL0;//保存低電平寬度
TH0=0;//定時器清0
TL0=0;//定時器清0
TR0=1;//開啟定時器T0
while(IR==1)//如果是高電平就等待
;
TR0=0;//關閉定時器T0
HighTime=TH0*256+TL0;//保存高電平寬度
if((LowTime<370)||(LowTime>640))
return0;//如果低電平長度不在合理范圍,則認為出錯,停止解碼
if((HighTime>420)&&(HighTime<620))//如果高電平時間在560微秒左右,即計數560/1.085=516次
temp=temp&0x7f;//(520-100=420,520+100=620),則該位是0
if((HighTime>1300)&&(HighTime<1800))//如果高電平時間在1680微秒左右,即計數1680/1.085=1548次
temp=temp|0x80;//(1550-250=1300,1550+250=1800),則該位是1
}
a[i]=temp;//將解碼出的位元組值儲存在a[i]
}
if(a[2]=~a[3])//驗證鍵數據碼和其反碼是否相等,一般情況下不必驗證用戶碼
return1;//解碼正確,返回1
}
/*------------------二進制碼轉換為壓縮型BCD碼,並顯示---------------*/
voidtwo_2_bcd(unsignedchardate)
{
unsignedchartemp;
temp=date;
date&=0xf0;
date>>=4;//右移四位得到高四位碼
date&=0x0f;//與0x0f想與確保高四位為0
if(date<=0x09)
{
WriteData(0x30+date);//lcd顯示鍵值高四位
}
else
{
date=date-0x09;
WriteData(0x40+date);
}
date=temp;
date&=0x0f;
if(date<=0x09)
{
WriteData(0x30+date);//lcd顯示低四位值
}
else
{
date=date-0x09;
WriteData(0x40+date);
}
WriteData(0x48);//顯示字元'H'
}
/************************************************************
函數功能:1602LCD顯示
*************************************************************/
voidDisp(void)
{
WriteAddress(0x40);//設置顯示位置為第一行的第1個字
two_2_bcd(a[0]);
WriteData(0x20);
two_2_bcd(a[1]);
WriteData(0x20);
two_2_bcd(a[2]);
WriteData(0x20);
two_2_bcd(a[3]);
}
/************************************************************
函數功能:主函數
*************************************************************/
voidmain()
{
unsignedchari;
LcdInitiate();//調用LCD初始化函數
delay(10);
WriteInstruction(0x01);//清顯示:清屏幕指令
WriteAddress(0x00);//設置顯示位置為第一行的第1個字
i=0;
while(string[i]!='