专注电子技术学习与研究
当前位置:单片机教程网 >> MCU设计实例 >> 浏览文章

16键计算器 c51实现

作者:佚名   来源:本站原创   点击数:  更新时间:2012年03月20日   【字体:

1、定时显示

  开辟一显示缓冲区,显示中断程序定时读取缓冲区显示各灯位,每灯位显示2.5ms,显示4只灯需要10ms,也就是显示的刷新频率是100Hz,亮度为25%。这样既保证亮度又不闪烁,同时CPU还有足够时间做其他事情。

2、键盘中断扫描

  利用键盘中断扫描程序,读出按键的键码,存入键盘缓冲区,供主程序读区。

3、计算器的主程序读取键盘缓冲区的键码,驱动计算器的运行,计算器的运行过程要清晰明了;

4、计算中需要十进制与十六进制(或二进制)转化。



 

#define LED_seg      XBYTE[0x6000]     //段码地址

#define LED_light    XBYTE[0x8000]      //灯位地址

#include "reg51.h"

#include "absacc.h"

unsigned char read_key(unsigned char);

unsigned char display(unsigned char,unsigned char);

void delay(unsigned int);

unsigned char light[4]={0xfe,0xfd,0xfb,0xf7};   //扫描值 灯位码

unsigned char seg[16]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,

0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; //段码

unsigned char disp_buffer[5]={1,0,0,0,0};            //当前灯位、 显示区

unsigned char key_buffer[2]={0,0xff};           //缓冲区满标志、键码

void main()

{

 Unsigned char temp[5];

 unsigned char i;

 unsigned char operator=0;

 unsigned int tempa,tempb;

 bit first=1;

 TMOD=0x01;       //置T0为方式1

 TL0=0x0;          //延时2.5mS的定时器初始值

 TH0=0xf7;          //赋初值

 PT0=1;             //定时中断0优先设置

TR0=1;           //启动定时

 ET0=1;           //允许定时

 IT1=0;        //电平触发低电平有效

 EX1=1;       //允许外中断

 PX1=1;       // 外部中断1优先级设定

 EA=1;        //中断总允许

 while(1)

      {

       P1=0xf0;                           //扫描

       if (key_buffer[0])

                {

                    switch(key_buffer[1])              //缓冲区满

                      {

                          case 0:

                          case 1:

                          case 2:

                          case 3:

                          case 4:

                          case 5:

                          case 6:

                          case 7:

                          case 8:

                          case 9: if (first)  for(i=1;i<=4;i++)   //保存数据,等待另一个数据输入

                                             {

                                                                temp[i]=disp_buffer[i];

                                                                disp_buffer[i]=0;                                           

                                                                }

                   first=0;

                                      for(i=1;i<=3;i++)  disp_buffer[i]=disp_buffer[i+1];

                                      disp_buffer[4]=key_buffer[1];

                                break;

                          case 10:

                          case 11:

                          case 12:

                          case 13:

                          case 14: first=1;

tempa=1000*disp_buffer[1]+100*disp_buffer[2]+10*disp_buffer[3]+disp_buffer[4];

                     tempb=1000*temp[1]+100*temp[2]+10*temp[3]+temp[4];

                     switch(operator)

                          {

                           case 10:tempa+=tempb;

                                   break;

                           case 11:tempa=tempb-tempa;

                                   break;

                           case 12:tempa*=tempb;

                                   break;

                           case 13:tempa=tempb/tempa;

                           }

                     tempa%=10000;

                     disp_buffer[1]=tempa/1000;

                     tempa%=1000;

                     disp_buffer[2]=tempa/100;

                     tempa%=100;

                     disp_buffer[3]=tempa/10;

                     disp_buffer[4]=tempa%10;

                                  operator=key_buffer[1];

                                        break;

                          case 15: for(i=1;i<=4;i++)  disp_buffer[i]=temp[i]=0;

                           first=1;

                                          operator=0;

                          }

                    key_buffer[0]=0;                     //缓冲区空

                    }          

       }

 }

void disp_LED() interrupt 1

{

 TL0=0x0;         //延时2.5mS的定时器初始值

 TH0=0xf7;

 if (disp_buffer[0]==5)  disp_buffer[0]=1;

 display(disp_buffer[disp_buffer[0]],disp_buffer[0]);

 disp_buffer[0]++;

 }

void get_keypad_code() interrupt 2

{

 unsigned char i,key;

 EA=0;

 for (i=0;i<=3;i++)

      {

       key=read_key(light[i]);    //读键码

       if (key!=0xff) delay(10);                //延时5-10ms,去抖动

       if (read_key(light[i])!=key) key=0xff;    //读键码比较

       if (key!=0xff)

                {

                    key_buffer[0]=1;                //缓冲区满

                    key_buffer[1]=key;      //键码

                    }                     

       }

 EA=1;

 }

unsigned char read_key(unsigned char scan)

{

 P1=scan;       //扫描

 switch(P1)     //返回扫描码->键码

    {

     case 0x77:return(0x7);

     case 0xb7:return(0x8);

     case 0xd7:return(0x9);

     case 0xe7:return(0xd);

     case 0x7b:return(0x4);

     case 0xbb:return(0x5);

     case 0xdb:return(0x6);

     case 0xeb:return(0xc);

     case 0x7d:return(0x1);

     case 0xbd:return(0x2);

     case 0xdd:return(0x3);

     case 0xed:return(0xb);

     case 0x7e:return(0xf);

     case 0xbe:return(0x0);

     case 0xde:return(0xe);

     case 0xee:return(0xa);

     default:  return(0xff);

     }

 }

unsigned char display(unsigned char disp_key,unsigned char n) //disp_key 显示字符,n 灯位

{

 if ((n<=4)&&(n>=1)) LED_light=light[n-1];

 else  LED_light=0xff;

 if ((disp_key>=0)&&(disp_key<=16)) LED_seg=seg[disp_key];

 else  LED_seg=0x00;

 return light[n-1];

 }

void delay(unsigned int k)     //延时k*1ms

{

unsigned int i,j;

 for (j=0;j<=k;j++)

    for(i=0;i<=1085;i++);

}

 

关闭窗口

相关文章