找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5269|回复: 0
收起左侧

51单片机加法器的制作

[复制链接]
ID:162044 发表于 2017-1-16 16:39 | 显示全部楼层 |阅读模式
完成数码管显示,按键相加的要求
此程序的电路图下载:http://www.51hei.com/f/ks51.pdf   只需要看数码管部分即可,其他部分可忽略掉.

代码如下

  1. #include<reg52.h>

  2. sbit ADDR0 = P1^0;
  3. sbit ADDR1 = P1^1;
  4. sbit ADDR2 = P1^2;
  5. sbit ADDR3 = P1^3;
  6. sbit ENLED = P1^4;

  7. sbit KEY_IN_1 = P2^4;
  8. sbit KEY_IN_2 = P2^5;
  9. sbit KEY_IN_3 = P2^6;
  10. sbit KEY_IN_4 = P2^7;
  11. sbit KEY_OUT_1 = P2^3;
  12. sbit KEY_OUT_2 = P2^2;
  13. sbit KEY_OUT_3 = P2^1;
  14. sbit KEY_OUT_4 = P2^0;

  15. unsigned char code LedChar[]={
  16.         0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,
  17.         0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
  18.         };
  19. unsigned char KeySta[4][4] = {
  20.         {1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1}
  21.         };
  22. unsigned char LedBuff[6] = {
  23.         0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
  24.         };
  25. unsigned char code KeyCodeMap[4][4] = { //矩阵按键编号到标准键盘键码的映射表
  26.     { 0x31, 0x32, 0x33, 0x26 }, //数字键1、数字键2、数字键3、向上键
  27.     { 0x34, 0x35, 0x36, 0x25 }, //数字键4、数字键5、数字键6、向左键
  28.     { 0x37, 0x38, 0x39, 0x28 }, //数字键7、数字键8、数字键9、向下键
  29.     { 0x30, 0x1B, 0x0D, 0x27 }  //数字键0、ESC键、  回车键、 向右键
  30.         };
  31. void  KeyDriver();
  32. void main()
  33. {
  34.     EA = 1;       //使能总中断
  35.     ENLED = 0;    //选择数码管进行显示
  36.     ADDR3 = 1;
  37.     TMOD = 0x01;  //设置T0为模式1
  38.     TH0  = 0xFC;  //为T0赋初值0xFC67,定时1ms
  39.     TL0  = 0x67;
  40.     ET0  = 1;     //使能T0中断
  41.     TR0  = 1;     //启动T0
  42.     LedBuff[0] = LedChar[0];  //上电显示0
  43.         
  44.     while (1)
  45.     {
  46.         KeyDriver();   //调用按键驱动函数
  47.     }
  48. }

  49. void ShowNumber(unsigned long num)
  50. {
  51.         signed char i;
  52.         unsigned char buf[6];

  53.         for(i=0; i<6; i++)
  54.         {
  55.                 buf[i] = num % 10;
  56.                 num = num / 10;
  57.         }

  58.         for(i=5; i>=1; i--)  \\gaoweibuxianshi
  59.         {
  60.                 if(buf[i] == 0)
  61.                 {
  62.                         LedBuff[i] = 0xFF;
  63.                 }
  64.                 else
  65.                         break;
  66.         }
  67.         
  68.         for(; i>=0; i--) \\
  69.         {
  70.                 LedBuff[i] = LedChar[buf[i]];        
  71.         }

  72. }
  73. void KeyAction(unsigned char keycode)
  74. {
  75.         static unsigned long result = 0;
  76.         static unsigned long addend = 0;
  77.         
  78.         if((keycode >= 0x30) &&(keycode <= 0x39))
  79.         {
  80.                 addend = (addend *10) + (keycode - 0x30);   
  81.                 ShowNumber(addend);
  82.         }
  83.         
  84.         else if(keycode == 0x26)
  85.         {
  86.                 result += addend;
  87.                 addend = 0;
  88.                 ShowNumber(result);
  89.         }
  90.         else if(keycode == 0x0D)
  91.         {
  92.                 result += addend;
  93.                 addend = 0;
  94.                 ShowNumber(result);
  95.         }
  96.         else if(keycode == 0x1B)
  97.         {
  98.                 addend = 0;
  99.                 result = 0;
  100.                 ShowNumber(addend);         
  101.         }        
  102.          
  103. }
  104. void  KeyDriver()
  105. {
  106.         unsigned char i, j;
  107.         static        unsigned char backup [4][4] = {
  108.         {1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1}
  109.         };

  110.         for(i=0; i<4; i++)
  111.                 {
  112.                         for(j=0; j<4; j++)
  113.                         {
  114.                                 if(backup[i][j] != KeySta[i][j])
  115.                                 {
  116.                                         if(backup[i][j] == 0)
  117.                                         {
  118.                                                 KeyAction(KeyCodeMap[i][j]);
  119.                                         }
  120.                                         backup[i][j] = KeySta[i][j];
  121.                                 }
  122.                         }        
  123.                 }

  124.         
  125. }

  126. /* 按键扫描函数,需在定时中断中调用,推荐调用间隔1ms */
  127. void KeyScan()
  128. {
  129.     unsigned char i;
  130.     static unsigned char keyout = 0;   //矩阵按键扫描输出索引
  131.     static unsigned char keybuf[4][4] = {  //矩阵按键扫描缓冲区
  132.         {0xFF, 0xFF, 0xFF, 0xFF},  {0xFF, 0xFF, 0xFF, 0xFF},
  133.         {0xFF, 0xFF, 0xFF, 0xFF},  {0xFF, 0xFF, 0xFF, 0xFF}
  134.     };

  135.     //将一行的4个按键值移入缓冲区
  136.     keybuf[keyout][0] = (keybuf[keyout][0] << 1) | KEY_IN_1;
  137.     keybuf[keyout][1] = (keybuf[keyout][1] << 1) | KEY_IN_2;
  138.     keybuf[keyout][2] = (keybuf[keyout][2] << 1) | KEY_IN_3;
  139.     keybuf[keyout][3] = (keybuf[keyout][3] << 1) | KEY_IN_4;
  140.     //消抖后更新按键状态
  141.     for (i=0; i<4; i++)  //每行4个按键,所以循环4次
  142.     {
  143.         if ((keybuf[keyout][i] & 0x0F) == 0x00)
  144.         {   //连续4次扫描值为0,即4*4ms内都是按下状态时,可认为按键已稳定的按下
  145.             KeySta[keyout][i] = 0;
  146.         }
  147.         else if ((keybuf[keyout][i] & 0x0F) == 0x0F)
  148.         {   //连续4次扫描值为1,即4*4ms内都是弹起状态时,可认为按键已稳定的弹起
  149.             KeySta[keyout][i] = 1;
  150.         }
  151.     }
  152.     //执行下一次的扫描输出
  153.     keyout++;                //输出索引递增
  154.     keyout = keyout & 0x03;  //索引值加到4即归零
  155.     switch (keyout)          //根据索引,释放当前输出引脚,拉低下次的输出引脚
  156.     {
  157.         case 0: KEY_OUT_4 = 1; KEY_OUT_1 = 0; break;
  158.         case 1: KEY_OUT_1 = 1; KEY_OUT_2 = 0; break;
  159.         case 2: KEY_OUT_2 = 1; KEY_OUT_3 = 0; break;
  160.         case 3: KEY_OUT_3 = 1; KEY_OUT_4 = 0; break;
  161.         default: break;
  162.     }
  163. }
  164. /* 数码管动态扫描刷新函数,需在定时中断中调用 */
  165. void LedScan()
  166. {
  167.     static unsigned char i = 0;  //动态扫描的索引

  168.     P0 = 0xFF;   //显示消隐
  169.     switch (i)
  170.     {
  171.         case 0: ADDR2=0; ADDR1=0; ADDR0=0; i++; P0=LedBuff[0]; break;
  172.         case 1: ADDR2=0; ADDR1=0; ADDR0=1; i++; P0=LedBuff[1]; break;
  173.         case 2: ADDR2=0; ADDR1=1; ADDR0=0; i++; P0=LedBuff[2]; break;
  174.         case 3: ADDR2=0; ADDR1=1; ADDR0=1; i++; P0=LedBuff[3]; break;
  175.         case 4: ADDR2=1; ADDR1=0; ADDR0=0; i++; P0=LedBuff[4]; break;
  176.         case 5: ADDR2=1; ADDR1=0; ADDR0=1; i=0; P0=LedBuff[5]; break;
  177.         default: break;
  178.     }
  179. }

  180. /* T0中断服务函数,用于数码管显示扫描与按键扫描 */
  181. void InterruptTimer0() interrupt 1
  182. {
  183.         TH0 = 0xFC;
  184.         TL0 = 0x67;
  185.         LedScan();
  186.         KeyScan();
  187. }
复制代码
回复

使用道具 举报

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

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

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