EDABOSS电子论坛

 找回密码
 立即注册
搜索
热搜: 活动 交友 discuz
查看: 630|回复: 2

[资料共享] 51单片机的红外计算器程序源码

[复制链接]

15

主题

0

回帖

84

E币

技术员

Rank: 2

积分
30
发表于 2018-8-4 20:39:40 | 显示全部楼层 |阅读模式
lcd1602和单片机接口自己在程序中看
单片机源程序如下:
  1. #include<reg52.h>    //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义
  2. #include<intrins.h>
  3. sbit RS = P1^7;   //定义端口
  4. sbit RW = P1^6;
  5. sbit EN = P1^5;
  6. sbit datk=P1^2;
  7. sbit rest=P1^3;
  8. sbit IR=P3^2;//中断0,红外接口                                                                                                                                                                                                                                                                                     //红外接口标志
  9. #define RS_CLR RS=0          //选择指令寄存器
  10. #define RS_SET RS=1          //选择数据寄存器
  11. #define RW_CLR RW=0          //把单片机数据写入lcd
  12. #define RW_SET RW=1          // 把lcm的数据读出到单片机
  13. #define EN_CLR EN=0          // 允许对lcm进行读写操作
  14. #define EN_SET EN=1          // 禁止对lcm进行读写操作
  15. #define DataPort P2 //定义数据端口 程序中遇到DataPort 则用P2 替换
  16. /*------------------------------------------------
  17.                 全局变量声明
  18. ------------------------------------------------*/
  19. unsigned int p=0,flag1,qingchu;
  20. static int a; //静态变量a
  21. unsigned char table1[4];
  22. unsigned char  irtime;//红外用全局变量

  23. bit irpro_ok,irok;
  24. unsigned char IRcord[4];     //四组数据,
  25. unsigned char irdata[33];        //前导码加32位数据
  26. unsigned char dat[] ={'0','1','2','3','4','5','6','7','8','9'};
  27. /*------------------------------------------------
  28.                   函数声明
  29. ------------------------------------------------*/

  30. void Ir_work(void);          //红外键值散转函数
  31. void Ircordpro(void);  //红外码值处理
  32. void Delay10ms();                //延时函数
  33. void delay1ms(void);           //延时函数
  34. void music(z);                        //语音函数
  35. /*------------------------------------------------
  36. uS延时函数,含有输入参数 unsigned char t,无返回值
  37. unsigned char 是定义无符号字符变量,其值的范围是
  38. 0~255 这里使用晶振12M,精确延时请使用汇编,大致延时
  39. 长度如下 T=tx2+5 uS
  40. ------------------------------------------------*/
  41. void DelayUs2x(unsigned char t)
  42. {
  43. while(--t);
  44. }
  45. /*------------------------------------------------
  46. mS延时函数,含有输入参数 unsigned char t,无返回值
  47. unsigned char 是定义无符号字符变量,其值的范围是
  48. 0~255 这里使用晶振12M,精确延时请使用汇编
  49. ------------------------------------------------*/
  50. void DelayMs(unsigned char t)
  51. {

  52. while(t--)
  53. {
  54.      //大致延时1mS
  55.      DelayUs2x(245);
  56.          DelayUs2x(245);
  57. }
  58. }
  59. /*------------------------------------------------
  60.               判忙函数
  61. ------------------------------------------------*/
  62. bit LCD_Check_Busy(void)
  63. {
  64. DataPort= 0xFF;
  65. RS_CLR;     //选择指令寄存器
  66. RW_SET;         //把lcm的数据读出到单片机
  67. EN_CLR;         //禁止对lcm进行读写操作
  68. _nop_();
  69. EN_SET;
  70. return (bit)(DataPort & 0x80);
  71. }
  72. /*------------------------------------------------
  73.               写入命令函数
  74. ------------------------------------------------*/
  75. void LCD_Write_Com(unsigned char com)
  76. {
  77. while(LCD_Check_Busy()); //忙则等待
  78. RS_CLR;
  79. RW_CLR;
  80. EN_SET;
  81. DataPort= com;
  82. _nop_();
  83. EN_CLR;
  84. }
  85. /*------------------------------------------------
  86.               写入数据函数
  87. ------------------------------------------------*/
  88. void LCD_Write_Data(unsigned char Data)
  89. {
  90. while(LCD_Check_Busy()); //忙则等待
  91. RS_SET;
  92. RW_CLR;
  93. EN_SET;
  94. DataPort= Data;
  95. _nop_();
  96. EN_CLR;
  97. }

  98. /*------------------------------------------------
  99.                 清屏函数
  100. ------------------------------------------------*/
  101. void LCD_Clear(void)
  102. {
  103. LCD_Write_Com(0x01);
  104. DelayMs(5);
  105. }

  106. /*------------------------------------------------
  107.               初始化函数
  108. ------------------------------------------------*/
  109. void LCD_Init(void)
  110. {
  111.    LCD_Write_Com(0x38);    /*显示模式设置*/
  112.    DelayMs(5);
  113.    LCD_Write_Com(0x38);
  114.    DelayMs(5);
  115.    LCD_Write_Com(0x38);
  116.    DelayMs(5);
  117.    LCD_Write_Com(0x38);
  118.    LCD_Write_Com(0x08);    /*显示关闭*/
  119.    LCD_Write_Com(0x01);    /*显示清屏*/
  120.    LCD_Write_Com(0x06);    /*显示光标移动设置*/
  121.    DelayMs(5);
  122.    LCD_Write_Com(0x0C);    /*显示开及光标设置*/
  123.    }

  124. /*------------------------------------------------
  125.                   定时器0中断处理
  126. ------------------------------------------------*/

  127. void tim0_isr (void) interrupt 1 using 1
  128. {
  129.   irtime++;  //用于计数2个下降沿之间的时间
  130. }

  131. /*------------------------------------------------
  132.                   外部中断0中断处理
  133. ------------------------------------------------*/
  134. void EX0_ISR (void) interrupt 0 //外部中断0服务函数
  135. {
  136.   static unsigned char  i;             //接收红外信号处理
  137.   static bit startflag;                //是否开始处理标志位

  138. if(startflag)
  139.    {
  140.     if(irtime<63&&irtime>=33)//引导码 TC9012的头码,9ms+4.5ms
  141.                         i=0;
  142.                     irdata[i]=irtime;//存储每个电平的持续时间,用于以后判断是0还是1
  143.                     irtime=0;
  144.                  i++;
  145.                             if(i==33)
  146.                               {
  147.                                    irok=1;
  148.                                  i=0;
  149.                                   }
  150.           }
  151.            else
  152.                 {
  153.                 irtime=0;
  154.                 startflag=1;
  155.                 }

  156. }



  157. void music(int z)
  158. {
  159. rest=1;
  160. delay1ms();
  161. delay1ms();
  162. rest=0;
  163. delay1ms();
  164. delay1ms();
  165.             //前面给语音复位,详情看语音资料
  166. while(z>0) //给b 个脉冲让语音播放b段语音
  167. {
  168. datk=1;
  169. delay1ms();
  170. datk=0;
  171. delay1ms();
  172. z--;
  173. }         }
  174. //************************************************************
  175. void delay1ms()
  176. {
  177.          unsigned char iii, jjj;

  178.         _nop_();
  179.         iii = 2;
  180.         jjj = 199;
  181.         do
  182.         {
  183.                 while (--jjj);
  184.         } while (--iii);
  185. }
  186. void delay(void)   //误差 -0.000000000227us
  187. {
  188.     unsigned char a,b,c;
  189.     for(c=13;c>0;c--)
  190.         for(b=247;b>0;b--)
  191.             for(a=142;a>0;a--);
  192.     _nop_;  //if Keil,require use intrins.h
  193. }



  194. /*------------------------------------------------
  195.                 定时器0初始化
  196. ------------------------------------------------*/
  197. void TIM0init(void)//定时器0初始化
  198. {

  199.   TMOD=0x02;//定时器0工作方式2,TH0是重装值,TL0是初值
  200.   TH0=0x00; //重载值
  201.   TL0=0x00; //初始化值
  202.   ET0=1;    //开中断
  203.   TR0=1;
  204. }        
  205. /*------------------------------------------------
  206.                   外部中断0初始化
  207. ------------------------------------------------*/
  208. void EX0init(void)
  209. {
  210. IT0 = 1;   //指定外部中断0下降沿触发,INT0 (P3.2)
  211. EX0 = 1;   //使能外部中断
  212. EA = 1;    //开总中断
  213.         
  214. }
  215. /*------------------------------------------------
  216.                   键值处理
  217. ------------------------------------------------*/
  218. int display(int j)
  219. {
  220.         int k;
  221.         switch(j)//判断第三个数码值
  222.                  {
  223.                                                   
  224.                          case 0x16:LCD_Write_Com(0x80+a);LCD_Write_Data('0');k=0; a++; break;//0   
  225.                          case 0x0c:LCD_Write_Com(0x80+a);LCD_Write_Data('1');k=1; a++; break;//1 显示相应的按键值
  226.                          case 0x18:LCD_Write_Com(0x80+a);LCD_Write_Data('2');k=2; a++;break;//2
  227.                          case 0x5e:LCD_Write_Com(0x80+a);LCD_Write_Data('3');k=3; a++;break;//3
  228.                          case 0x08:LCD_Write_Com(0x80+a);LCD_Write_Data('4');k=4; a++;break;//4         
  229.                          case 0x1c:LCD_Write_Com(0x80+a);LCD_Write_Data('5');k=5; a++;break;//5  
  230.                          case 0x5a:LCD_Write_Com(0x80+a);LCD_Write_Data('6');k=6; a++;break;//6 显示相应的按键值
  231.                          case 0x42:LCD_Write_Com(0x80+a);LCD_Write_Data('7');k=7; a++;break;//7
  232.                          case 0x52:LCD_Write_Com(0x80+a);LCD_Write_Data('8');k=8; a++;break;//8
  233.                          case 0x4a:LCD_Write_Com(0x80+a);LCD_Write_Data('9');k=9; a++;break;//9
  234.            }

  235.         return k;
  236. }

  237. void Ir_work(void)//红外键值散转程序
  238. {
  239.         
  240.          int k=0,flag1,count1,count2,count3,count4;
  241.          int table1[6];
  242.          int count;
  243.          if(IRcord[2]!=0x09&&IRcord[2]!=0x44&&IRcord[2]!=0x15&&IRcord[2]!=0x40&&IRcord[2]!=0x43&&IRcord[2]!=0x07)
  244.          {
  245.          k=display(IRcord[2]);
  246.          table1[p++]=k;
  247.          }
  248.         else  if(IRcord[2]==0x09||IRcord[2]==0x44||IRcord[2]==0x15||IRcord[2]==0x40)
  249.          {
  250.                  switch(IRcord[2])
  251.                 {
  252.                           case 0x09:LCD_Write_Com(0x80+a);LCD_Write_Data('+');flag1=1;a++;break;//+  
  253.                          case 0x15:LCD_Write_Com(0x80+a);LCD_Write_Data('-');flag1=2;a++;break;//-显示相应的按键值
  254.                          case 0x44:LCD_Write_Com(0x80+a);LCD_Write_Data('*');flag1=3;a++;break;//*
  255.                          case 0x40:LCD_Write_Com(0x80+a);LCD_Write_Data('/');flag1=4;a++;break;///
  256.                         //case  0x10:
  257.                 }
  258.          }
  259.         else if(IRcord[2]==0x07)
  260.          {
  261.                          LCD_Write_Com(0x80+a);
  262.                         LCD_Write_Data('=');
  263.                          switch(flag1)
  264.                         {
  265.                                 case 1:        count=table1[0]*10+table1[1]+table1[2]*10+table1[3];
  266.                                 if(count<100)
  267.                                 {
  268.                                         count1=count/10;
  269.                                         count2=count%10;  
  270.                                         LCD_Write_Com(0x80+a+1);
  271.                                         LCD_Write_Data(dat[count1]);

  272.                                         LCD_Write_Com(0x80+a+2);
  273.                                         LCD_Write_Data(dat[count2]);
  274.                                         delay();
  275.                                         music(count1+1);
  276.                                         delay();
  277.                                 music(count2+1);
  278.                                         delay();
  279.                                 }
  280.                                 else if(count>=100)
  281.                                 {
  282.                                         count1=count/100;         LCD_Write_Com(0x80+a+1);LCD_Write_Data(dat[count1]);
  283.                                         count2=count/10%10;         LCD_Write_Com(0x80+a+2);LCD_Write_Data(dat[count2]);
  284.                                         count3=count%10;         LCD_Write_Com(0x80+a+3);LCD_Write_Data(dat[count3]);
  285.                                         delay();
  286.                                         music(count1+1);
  287.                                         delay();
  288.                                         music(count2+1);
  289.                                         delay();
  290.                                         music(count3+1);
  291.                                         delay();
  292.                                 }
  293.                                                                 break;
  294.                                 case 2:          count=table1[0]*10+table1[1]-(table1[2]*10+table1[3]);
  295.                                                 count1=count/10;           LCD_Write_Com(0x80+a+1);LCD_Write_Data(dat[count1]);
  296.                                                 count2=count%10;        LCD_Write_Com(0x80+a+2); LCD_Write_Data(dat[count2]);
  297.                                                 music(count1+1);
  298.                                            delay();
  299.                                            music(count2+1);
  300.                                            delay();
  301.                                                 break;
  302.                                        
  303.                                 
  304.                                 case 3:count=(table1[0]*10+table1[1])*(table1[2]*10+table1[3]);
  305.                                         if(count<10) { LCD_Write_Com(0x80+a+1);LCD_Write_Data(dat[count]);}
  306.                                         else if(count<100)
  307.                                         {
  308.                                                 count1=count/10;  LCD_Write_Com(0x80+a+1);        LCD_Write_Data(dat[count1]);
  309.                                                 count2=count%10;   LCD_Write_Com(0x80+a+2);        LCD_Write_Data(dat[count2]);
  310.                                             delay();
  311.                                                 music(count1+1);
  312.                                             delay();
  313.                                                 music(count2+1);
  314.                                             delay();
  315.                                         }
  316.                                        
  317.                                         else if(count<1000)
  318.                                         {
  319.                                                 count1=count/100;        LCD_Write_Com(0x80+a+1);        LCD_Write_Data(dat[count1]);
  320.                                                 count2=count/10%10;          LCD_Write_Com(0x80+a+2);        LCD_Write_Data(dat[count2]);
  321.                                                 count3=count%10;          LCD_Write_Com(0x80+a+3);        LCD_Write_Data(dat[count3]);
  322.                                                 delay();
  323.                                                 music(count1+1);
  324.                                             delay();
  325.                                                 music(count2+1);
  326.                                             delay();
  327.                                                 music(count3+1);
  328.                                             delay();
  329.                                         }
  330.                                         else   if(count<10000)
  331.                                          {
  332.                                                   count1=count/1000;                   LCD_Write_Com(0x80+a+1);        LCD_Write_Data(dat[count1]);
  333.                                                 count2=count/100%10;          LCD_Write_Com(0x80+a+2);        LCD_Write_Data(dat[count2]);
  334.                                                 count3=count/10%10;                   LCD_Write_Com(0x80+a+3);        LCD_Write_Data(dat[count3]);
  335.                                                 count4=count%10;                 LCD_Write_Com(0x80+a+4);        LCD_Write_Data(dat[count4]);
  336.                                             delay();
  337.                                                 music(count1+1);
  338.                                             delay();
  339.                                                 music(count2+1);
  340.                                             delay();
  341.                                                 music(count3+1);
  342.                                             delay();
  343.                                             music(count4+1);
  344.                                             delay();
  345.                                          }
  346.                                                         break;
  347.                                 case 4:                count=(table1[0]*10+table1[1])/(table1[2]*10+table1[3]);
  348.                         
  349.                                           
  350.                                  
  351.                                  if(count<10)
  352.                                                 {                                                              
  353.                                                   LCD_Write_Com(0x80+a+1);LCD_Write_Data('0');        //LCD_Write_Data(dat[count]);
  354.                                                   LCD_Write_Com(0x80+a+2);LCD_Write_Data(dat[count]);
  355.                                                   delay();
  356.                                                   music(1);
  357.                                               delay();
  358.                                                   music(count+1);
  359.                                               delay();
  360.                                                 }
  361.                                                 else  
  362.                                                 {
  363.                                                         count1=count/10;  LCD_Write_Com(0x80+a+1);        LCD_Write_Data(dat[count1]);
  364.                                                         count2=count%10;  LCD_Write_Com(0x80+a+2);        LCD_Write_Data(dat[count2]);
  365.                                                         delay();
  366.                                                     music(count1+1);
  367.                                                 delay();
  368.                                                     music(count2+1);
  369.                                                 delay();
  370.                                                 }           
  371.                                                 break;
  372.                         }
  373.                         flag1=0;
  374.          }
  375.          else
  376.          {
  377.          if(IRcord[2]==0x43)
  378.          {
  379.                 LCD_Write_Com(0x01);
  380.          qingchu=1;
  381.                  
  382.          }
  383. }        
  384.         
  385.                         
  386. }
  387. /*------------------------------------------------
  388.                 红外码值处理
  389. ------------------------------------------------*/
  390. void Ircordpro(void)//红外码值处理函数
  391. {
  392.   unsigned char i, j, k;
  393.   unsigned char cord,value;

  394.   k=1;
  395.   for(i=0;i<4;i++)      //处理4个字节
  396.      {
  397.       for(j=1;j<=8;j++) //处理1个字节8位
  398.          {
  399.           cord=irdata[k];
  400.           if(cord>7)//大于某值为1,这个和晶振有绝对关系,这里使用12M计算,此值可以有一定误差
  401.              value|=0x80;
  402.           if(j<8)
  403.                     {
  404.                          value>>=1;
  405.                         }
  406.            k++;
  407.          }
  408.      IRcord[i]=value;
  409.      value=0;
  410.      }
  411.          irpro_ok=1;//处理完毕标志位置1
  412. }

  413. /*------------------------------------------------
  414.                     主函数
  415. ------------------------------------------------*/
  416. void main(void)
  417. {
  418. int m;
  419. EX0init(); //初始化外部中断
  420. TIM0init();//初始化定时器
  421. LCD_Init();
  422. LCD_Clear();//清??
  423.         
  424. while(1)//主循环
  425.    {
  426.     while(irok==0);                        //如果接收好了进行红外处理
  427.                  irok=0 ;
  428.            Ircordpro();
  429.           while(irpro_ok==0);                   //如果处理好后进行工作处理,如按对应的按键后显示对应的数字等
  430.                    irpro_ok=0;
  431.            Ir_work();
  432.            if(qingchu==1)
  433.            {
  434.                            qingchu=0;
  435.                         a=0;
  436.                  LCD_Write_Com(0x01);
  437.                  LCD_Write_Com(0x80);
  438.                         p=0;
  439.                  for(m=0;m<6;m++)
  440.                          table1[m]=0;
  441.            }
  442.         }
  443.         }


复制代码
程序.zip (48.59 KB, 下载次数: 3)
积分规则
回复

使用道具 举报

30

主题

33

回帖

39

E币

技术员

Rank: 2

积分
93
发表于 2018-10-22 10:37:09 | 显示全部楼层
谢谢分享
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

Archiver|手机版|小黑屋|EDABOSS电子论坛

GMT+8, 2024-4-24 08:14 , Processed in 0.041270 second(s), 23 queries .

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表