⑴ 紅外線遙控器解碼C語言程序代碼 SM0038接受頭
//////////////////////////////////////
//晶振頻率為6MHz  一個機器周期2us   //
//實現按鍵地址碼、指令碼的數碼管顯示//
//2010-06-01              //
//////////////////////////////////////
#include<reg52.h>
#define uchar unsigned char
#define uint unsigned int
sbit la=P2^6;
sbit wela=P2^7;
bit overflag,dataover;//開始接收數據,數據處理完畢
uchar timedata;//定時器0溢出次數
uchar chubus[33];//33Byte數據 timedata組成的數組
uchar jieguos[4];//地址碼,地址反碼,數據碼,數據反碼;
void initial()//初始化
{
 IT0=1;EX0=1;//負邊沿觸發
 
  TMOD=0x01;ET0=1;TR0=1;//模式1,十六位定時器
 TH0=0xff;TL0=0x06;//0.5ms溢出
 EA=1; //開總中斷
}
void time0() interrupt 1//定時器0中斷
{TH0=0xff;TL0=0x06;
 timedata++;
}
void ex0() interrupt 0//外部中斷0,接收數據
{
 static bit startflag;//開始接收
 static uchar i;
 
  if(startflag)
 {
  if(timedata<32&&timedata>=16) i=0;
  chubus[i]=timedata;
  timedata=0;
  i++;
  if(i==33){overflag=1;i=0;}
    
 } 
 else
 {
  startflag=1;
  timedata=0;
 }
}
void chulidata()
{
 uchar chubu;//初步數據
 uchar jieguo;//結果數據
 uchar x,y,z=1;
 for(x=0;x<4;x++)//處理四組數據
 {
  for(y=1;y<=8;y++)//處理一組數據8Byte
    {
   chubu=chubus[z];
   jieguo=jieguo>>1; 
   if(chubu>3) jieguo=jieguo|0x80;//大於1.5mS為1
   z++;
  }
 jieguos[x]=jieguo;
 jieguo=0;
 }
 dataover=1;
 
}
void delay(uint z)
{
 uint x ,y;
 for(x=z;x>0;x--)
  for(y=110;y>0;y--);
}
void display()
{
  uchar gao,gao1;
 uchar di,di1;
 uchar code table[]={
 0x3f,0x06,0x5b,0x4f,
 0x66,0x6d,0x7d,0x07,
 0x7f,0x6f,0x77,0x7c,
 0x39,0x5e,0x79,0x71};
 gao1=table[jieguos[0]/16];
 di1=table[jieguos[0]%16];
 gao=table[jieguos[2]/16];
 di=table[jieguos[2]%16];
  la=1;
  P0=gao1;
  la=0;
  P0=0xff; 
  wela=1;
  P0=0x7e;
  wela=0;
  delay(2);
la=1;
  P0=di1;
  la=0; 
  P0=0xff;
  wela=1;
  P0=0xfd;
  wela=0;
  delay(2);//地址碼
  la=1;
  P0=gao;
  la=0;
  P0=0xff; 
  wela=1;
  P0=0x77;
  wela=0;
  delay(2);
la=1;
  P0=di;
  la=0; 
  P0=0xff;
  wela=1;
  P0=0x6f;
  wela=0;
  delay(2);//指令碼
} 
void main()
{
  initial();//初始化
 while(1)
 {
  
  if(overflag)//數據接收完畢
  {
   chulidata();//處理數據,完成標志dataover
   overflag=0; 
  } 
  if(dataover)
  {
   display();//數碼管顯示
   }
  }
}
⑵ 基於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_();
  }
 }
}
⑶ 高分請教高人,紅外遙控解碼程序C語言,要求每一行都解釋為什麼什麼作用。非常感謝!!先給100分,好再加
#include <reg52.h>   //特殊寄存器頭文件  
#define c(x) (x*110592/120000) //是晶振值,為計數器計一下所需要的微秒數,120000為12M,110592為11.0592M
sbit Ir_Pin=P3^3;   //位聲明,把P3.3/外部中斷1的狀態讀到Ir_Pin中
unsigned char code Led_Tab[]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,
0xf8,0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};      //共陽極數碼顯示碼0-F.
unsigned char code Led_Sel[]={0xe,0xd,0xb,0x7};     //位選編碼
unsigned char Led_Buf[4];     //顯示緩沖區
char Led_Index;              //位選信號定義        
unsigned char Ir_Buf[4];     //用於保存解碼結果
//==============================================================
//數碼管掃描
timer0() interrupt 1 using 1  //定時器中斷零程序
{
TH0=(65536-1000)/256; 
TL0=(65536-1000)%256;        //定時器0設定約1000us中斷一次,用於數碼管掃描
P0=0xff;                     //數碼管初始顯示零
P2=Led_Sel[Led_Index];       //位選
P0=Led_Tab[Led_Buf[Led_Index]];   //段選
if(++Led_Index>3) Led_Index=0;    //四個掃描完了,到第一個數碼管
}
//==============================================================
unsigned int Ir_Get_Low()    //脈沖為低電平的時間
{
TL1=0;
TH1=0;                    //為定時器1賦初值
TR1=1;                   //開啟定時器1
while(!Ir_Pin && (TH1&0x80)==0);  //判斷,如果P3.3口為低電平則執行TR1=0               
TR1=0;                //關閉定時器1          
return TH1*256+TL1;         //返回TH1*256+TL1的值
}
//=============================================================
unsigned int Ir_Get_High()    //脈沖高電平時間
{
TL1=0;
TH1=0;                      //為定時器1賦初值
TR1=1;                      //開啟定時器1
while(Ir_Pin && (TH1&0x80)==0);    //判斷,如果P3.3口為低電平則執行TR1=0
TR1=0;                     //關閉定時器1
return TH1*256+TL1;               //返回TH1*256+TL1的值
}
//==============================================================
main()
{
unsigned int temp;
char i,j;
Led_Index=1;
TMOD=0x11;
TL0=(65536-1000)%256;
TH0=(65536-1000)/256;    //定時器0設定約1000us中斷一次,用於數碼管掃描
EA=1;                    //開總中斷
ET0=1;                   //定時計數器0的開放控制位
TR0=1;                   //定時器0的運行控制位
Led_Buf[0]=0;
Led_Buf[1]=0;
Led_Buf[2]=0;
Led_Buf[3]=0;            //顯示區設成0
do{
restart:
while(Ir_Pin);                  //判斷P3.3口
temp=Ir_Get_Low();              //取脈沖為低電平的時間 
if(temp<c(8500) || temp>c(9500)) continue;   //引導脈沖低電平9000
temp=Ir_Get_High();             //取脈沖高電平時間
if(temp<c(4000) || temp>c(5000)) continue;    //引導脈沖高電平4500
for(i=0;i<4;i++)                //4個位元組
    for(j=0;j<8;j++)            //每個位元組8位
    {
 temp=Ir_Get_Low();
 if(temp<c(200) || temp>c(800)) goto restart;  //根據編碼格式,低電平小於0.2ms大於0.8ms視為無效電平,重新檢測
 temp=Ir_Get_High();
 if(temp<c(200) || temp>c(2000)) goto restart; //根據編碼格式,低電平小於0.2ms大於2ms視為無效電平,重新檢測
 Ir_Buf[i]>>=1;                            //把Ir_Buf[i]右移一位,然後賦值給Ir_Buf[i]
 if(temp>c(1120)) Ir_Buf[i]|=0x80;         //根據編碼格式,如果電平大於1.12ms,則把0x80賦值給Ir_Buf[i]
} 
Led_Buf[0]=Ir_Buf[2]&0xf;
Led_Buf[1]=(Ir_Buf[2]/16)&0xf;
Led_Buf[2]=Ir_Buf[3]&0xf;
Led_Buf[3]=(Ir_Buf[3]/16)&0xf; //顯示結果
}while(1);
}
⑷ 紅外遙控器同一個鍵控制一個I/O口,第一次開,第二次關,比如P0^1開啟關閉C語言代碼高手請進。
第一,收到你要的按鍵後,是取反某個位,而不是賦值。如P10=~P10;(建議取個名字)
第二,為了避免一次按鍵,多次觸發,對按鍵值操作後,需要進行清除處理;(即賦值成無效的按鍵值),如 IrValue[2]=0xff;(如果沒有按鍵是 0xff的話)
第三,最正確的做法是,收到遙控後,額外提供一個標志位(如 IR_UpdateFlag),程序識別標志位後,進行按鍵選擇,處理完按鍵後,清除標志位。這樣可以避免程序重復、多餘執行按鍵選擇(雖然是無效的按鍵,但那也是選擇),可以提高系統執行效率!
