使用方法:打开串口调试助手,设置为9600 bps 单片机这边用11.0592MHz的晶振,使用sm0038或者其他型号的红外接收头按下面的电路连接好,其中out直接与单片机的p3.2脚相连.按下遥控器,串口调试助手便会出现解码值.

/******************************************************************/
 /*   本程序的蓝本从网上搜集,经修改并注释,万能遥控器解码成功      */
 /*    晶振:11.0592MHz                                     */ /*    整理与测试胡琴 2012.5.15    */
 /*************************   说 明 *********************************/
 /*    以一个9ms的低电平和4.5ms的高电平为引导码,后跟32位二进制代码 */
 /*    前16位为8位用户码及其反码,后16位为8位的操作码及其反码        */
 /*    以脉宽为0.565ms、间隔0.56ms、周期为1.125ms的组合表示"0";  */
 /*    以脉宽为0.565ms、间隔1.685ms、周期为2.25ms的组合表示"1"。  */
 /*    注意:接收码的脉宽与间隔是对发射码取反的,即间隔是0.565ms    */
 /*    解码后共有四个十六进制码,本程序取第三个作为识别码             */
 /*******************************************************************/
 #include <reg52.h>
 #define uchar unsigned char
 uchar  data IRcode[4];   //定义一个4字节的数组用来存储代码
 uchar  CodeTemp;   //编码字节缓存变量
 uchar  i,j,k;    //延时用的循环变量
 sbit   IRsignal=P3^2;   //HS0038接收头OUT端直接连P3.2(INT0)
 sbit P0_0=P0^0;  //P0连接到 LED 上
 sbit P0_1=P0^1;
 sbit P0_2=P0^2;
 /**************************延时0.9ms子程序**********************/
 void Delay0_9ms(void)
 {
   uchar j,k;
   for(j=18;j>0;j--)
        for(k=20;k>0;k--)
           ;
 }
 /***************************延时1ms子程序**********************/
 void Delay1ms(void)
 {
   uchar i,j;
   for(i=2;i>0;i--)
        for(j=230;j>0;j--)
           ;
 }
 /***************************延时4.5ms子程序**********************/
 void Delay4_5ms(void)
 {
   uchar i,j;
   for(i=10;i>0;i--)
        for(j=225;j>0;j--)
           ;
 }
 /**************************** 延时子程序 ************************/
 void Delay(void)
 {
 uchar i,j,k;
 for(i=200;i>0;i--)  
        for(j=200;j>0;j--)
             for(k=3;k>0;k--)
          ;
 }
 /********************   中断0解码服务子程序  ********************/
 void int0(void) interrupt 0 using 2      
 {
   EA = 0;   //??? 可以这样,跳入中断,但仍可对P3.2(INT0)进行电平变化的读取
   for(k=0;k<10;k++)
   {
        Delay0_9ms();
        if (IRsignal==1)         //如果0.9ms后IRsignal=1,说明不是引导码
        {
           k=10;
     break;
     } 
    else if(k==9)           //如果 持续了10×0.9ms=9ms的低电平,说明是引导码
        {
           while(IRsignal==0);
              Delay4_5ms();        //跳过持续4.5ms的高电平
              for(i=0;i<4;i++)       //分别读取4个字节
              {
          for(j=1;j<=8;j++)      //每个字节8个bit的判断
                   {
             while(IRsignal==0);       //等待上升沿   此处用得很好:因为0.56ms的低电平(接收时)是代码0与1的相同部分
                         Delay0_9ms();    //从上升沿那一时刻开始延时0.9ms(因为0.9介于0.56(=1.125-0.56)与1.69(=2.25-0.56)之间),再判断IRsignal
                         if(IRsignal==1)     //如果IRsignal是"1",高位置"1",并向右移一位
                         {
             Delay1ms();   //为什么要延时1ms呢?因为要使IRsignal跳至低电平(即0.56ms的0与1相同部分上)
                               CodeTemp=CodeTemp|0x80;  //此处的算法很好
                               if(j<8) CodeTemp=CodeTemp>>1;
                         }
                         else
             if(j<8)
                CodeTemp=CodeTemp>>1;//如果IRsignal是"0",则直接向右移一位,自动补"0"
                   }
                   IRcode=CodeTemp;
                   CodeTemp=0;
              }
              for(i=0;i<4;i++)        //通过串口将代码发出
              {
                     SBUF=IRcode;
                     while(!TI);       //等待一个字节发送完毕
                     TI=0;
              }
              Delay();
        }
   }
   EA = 1;
 }
 /***********************串口初始化程序*********************/
 void initUart(void)
 {
   TMOD |= 0x20; //  
   SCON = 0x50; //
   PCON |= 0x80; //
   TH1 = 250;     // 9600 bps @ 11.0592MHz
   TL1 = 250;
   TR1 = 1;
 }
 /**************************主程序*************************/
 void main()
 {
   P0=0XFF;
   initUart();
   IT0 = 1;    //INT0为负边沿触发, (1:负边沿触发,0:低电平触发)
   EX0 = 1;    //外部中断INT0开, (1:开,      0:关    )
   EA = 1;     //开所有中断
   CodeTemp = 0;     //初始化红外编码字节缓存变量
   Delay();
   while(1)
   {
      switch(IRcode[2])
   {
         case 0x42:P0=0XFF;P0_0=0;break;
      case 0x4e:P0=0XFF;P0_1=0;break;
      case 0x52:P0=0XFF;P0_2=0;break;
   }
   }
 }