找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3191|回复: 10
打印 上一主题 下一主题
收起左侧

单片机188数码管消影程序调试问题

  [复制链接]
跳转到指定楼层
楼主
10黑币
大家好,请教一下,我在使用188数码管时,遇到了一个消影问题,数码管动态扫描,在刷新函数中,已经加入了消影语句,即各个引脚设置为高阻输入,但是不应该亮的数码管还是会有弱光,所以显示并不清晰,当然消影函数还是有一定的作用,不加消影函数,几乎无法分辨数字,请大家帮帮忙,看看是哪里有问题

硬件上:使用的是STC8G1K08系列,直接IO口驱动数码管
程序是从 188数码管显示 这里几乎不动的写过来的,刷新函数在定时器中断函数中每 4ms 刷新一次
  1. #include "188.h"
  2. #include "delay.h"
  3. #include "intrins.h"
  4. #include "stc8g.h"

  5. sbit PSEG_1 = P3^6;
  6. sbit PSEG_2 = P3^5;
  7. sbit PSEG_3 = P3^4;
  8. sbit PSEG_4 = P3^3;
  9. sbit PSEG_5 = P3^2;

  10. #define PIN1_IN() P3M1 |= 0x40;P3M0 &= 0xbf;                                                        //把单片机IO口配置为高阻输入
  11. #define PIN2_IN() P3M1 |= 0x20;P3M0 &= 0xdf;
  12. #define PIN3_IN() P3M1 |= 0x10;P3M0 &= 0xef;
  13. #define PIN4_IN() P3M1 |= 0x08;P3M0 &= 0xf7;
  14. #define PIN5_IN() P3M1 |= 0x04;P3M0 &= 0xfb;

  15. #define PIN1_L() P3M0 |= 0x40;P3M1 &= 0xbf;PSEG_1 = 0;   //配置为推挽输出,且拉低
  16. #define PIN2_L() P3M0 |= 0x20;P3M1 &= 0xdf;PSEG_2 = 0;
  17. #define PIN3_L() P3M0 |= 0x10;P3M1 &= 0xef;PSEG_3 = 0;
  18. #define PIN4_L() P3M0 |= 0x08;P3M1 &= 0xf7;PSEG_4 = 0;
  19. #define PIN5_L() P3M0 |= 0x04;P3M1 &= 0xfb;PSEG_5 = 0;

  20. #define PIN1_H() P3M0 |= 0x40;P3M1 &= 0xbf;PSEG_1 = 1;  //配置为推挽输出,且拉高
  21. #define PIN2_H() P3M0 |= 0x20;P3M1 &= 0xdf;PSEG_2 = 1;
  22. #define PIN3_H() P3M0 |= 0x10;P3M1 &= 0xef;PSEG_3 = 1;
  23. #define PIN4_H() P3M0 |= 0x08;P3M1 &= 0xf7;PSEG_4 = 1;
  24. #define PIN5_H() P3M0 |= 0x04;P3M1 &= 0xfb;PSEG_5 = 1;


  25. /************************************************************************************/
  26. void LED_GPIO_Init(void)//初始化IO口
  27. {  
  28.         P3PU &= 0x83;                        //关闭上拉电阻
  29.         P3M1 &= 0x83;           //设置P3.2~P3.6为推挽输出
  30.         P3M0 |= 0x7c;
  31. }
  32. /********************************熄灭所有数码管***********************************/
  33. void Set_AllPin_INPUT(void)
  34. {
  35.         PIN1_IN();//把单片机IO口配置为输入模式
  36.         PIN2_IN();
  37.         PIN3_IN();
  38.         PIN4_IN();
  39.         PIN5_IN();
  40. }

  41. unsigned char bai=0,shi=0,ge=0;//百位,十位,个位,
  42. unsigned short display_sram=0;
  43. unsigned short code Segment[3][11]={
  44.   {0x0000,0x0006,0x0000,0x0000,0x0000,0x0000,0x0000,0x0000,0x0000,0x0000,0x0000},//百位
  45.   {0x0770,0x0420,0x0741,0x0661,0x0431,0x0271,0x0371,0x0460,0x0771,0x0671,0x0000},//十位
  46.   {0xE888,0x8080,0xD808,0xD880,0xB080,0x7880,0x7888,0x8880,0xF888,0xF880,0x0000} //个位
  47.         };

  48. void Display_Scan1(void)
  49. {
  50.           PIN1_L();        //拉低Pin1

  51.     if(display_sram&0x8000)        //取出最高位
  52.         PIN2_H();
  53.     if(display_sram&0x4000)
  54.         PIN3_H();
  55.     if(display_sram&0x2000)
  56.         PIN4_H();        
  57.     if(display_sram&0x1000)
  58.         PIN5_H();
  59. }
  60. void Display_Scan2(void)
  61. {
  62.           PIN2_L();        //拉低Pin2
  63.         
  64.     if(display_sram&0x0800)
  65.         PIN1_H();
  66.     if(display_sram&0x0400)
  67.         PIN3_H();               
  68.     if(display_sram&0x0200)
  69.         PIN4_H();
  70.     if(display_sram&0x0100)
  71.         PIN5_H();
  72. }
  73. void Display_Scan3(void)
  74. {
  75.           PIN3_L();        //拉低Pin3
  76.         
  77.     if(display_sram&0x0080)
  78.         PIN1_H();
  79.     if(display_sram&0x0040)
  80.         PIN2_H();;
  81.     if(display_sram&0x0020)
  82.         PIN4_H();
  83.     if(display_sram&0x0010)
  84.         PIN5_H();
  85. }
  86. void Display_Scan4(void)
  87. {
  88.           PIN4_L();        //拉低Pin4
  89.         
  90.     if(display_sram&0x0008)
  91.         PIN1_H();
  92.     if(display_sram&0x0004)
  93.         PIN2_H();
  94.     if(display_sram&0x0002)
  95.         PIN3_H();
  96.     if(display_sram&0x0001)
  97.         PIN5_H();
  98. }

  99. /****************************显示函数***********************************/

  100. //放在定时器中断函数,5ms运行一次,扫描一轮需要4*5ms=20ms,
  101. //也就是50Hz的刷新频率,如果感觉闪烁,可以4ms运行。
  102. void Display_tube(void)
  103. {
  104.     static unsigned char case_cnt;//逐行扫描
  105.     Set_AllPin_INPUT();//消影作用
  106.     display_sram=0;
  107.     display_sram=Segment[0][bai] | Segment[1][shi] | Segment[2][ge];//显示百位,十位,个位
  108.     switch(case_cnt)
  109.     {
  110.       case 0x00:Display_Scan1();case_cnt++;break;
  111.       case 0x01:Display_Scan2();case_cnt++;break;
  112.       case 0x02:Display_Scan3();case_cnt++;break;
  113.       case 0x03:Display_Scan4();case_cnt=0;break;
  114.       default:case_cnt=0;break;
  115.     }
  116. }

  117. void Dispaly_Dig(unsigned int dig)//拆分
  118. {        
  119.         bai = dig/100;
  120.         shi = (dig/10)%10;
  121.         ge = dig%10;
  122.         Display_tube();
  123. }
复制代码

这是定时器中断函数,4ms一次刷新
  1. void TM1_Isr() interrupt 3 //定时器1
  2. {
  3.         static unsigned char case_cnt;//定时中断次数
  4.         case_cnt++;
  5.         if(case_cnt==4)
  6.         {
  7.                 case_cnt=0;
  8.                 Dispaly_Dig(12);//显示数字12
  9.         }
  10. }
复制代码

这是原理图,直接连到了单片机引脚上



这是效果图,图一加有消影函数,图二未添加,消影函数如图三所示



拜托大家了

最佳答案

查看完整内容

采用查理复用IO驱动方式,扫描周期小于1.25ms,每次只需点亮一个笔段,分16次完成一轮,不必消隐。
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏2 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:213173 发表于 2022-2-12 15:03 | 只看该作者
采用查理复用IO驱动方式,扫描周期小于1.25ms,每次只需点亮一个笔段,分16次完成一轮,不必消隐。 查理复用IO驱动数码管.docx (92.31 KB, 下载次数: 131)

回复

使用道具 举报

板凳
ID:313048 发表于 2022-2-12 16:22 | 只看该作者
先显示再消影,你先消影再显示就相当于程序跑了一圈,然后你开头灭一下 大部分时间都是亮的
回复

使用道具 举报

地板
ID:962396 发表于 2022-2-12 22:37 | 只看该作者
wulin 发表于 2022-2-12 21:10
采用查理复用IO驱动方式,扫描周期小于1.25ms,每次只需点亮一个笔段,分16次完成一轮,不必消隐。

谢谢学习一下
回复

使用道具 举报

5#
ID:575020 发表于 2022-2-13 23:40 | 只看该作者
以前还在纠结数码管用573、595、还有TM、MAX系列到底那种方式好,今天看到这种5脚2位半数码管又彻底傻眼了,想不通怎么会有如此怪异设计的数码管。

1.jpg (63.78 KB, 下载次数: 80)

1.jpg
回复

使用道具 举报

6#
ID:399179 发表于 2022-3-26 21:58 来自手机 | 只看该作者
哎呀!我也遇到这样的问题了
回复

使用道具 举报

7#
ID:1067504 发表于 2023-7-21 22:00 来自手机 | 只看该作者
能不能分享完整的源代码,我想学习一下,谢谢!
回复

使用道具 举报

8#
ID:1073939 发表于 2023-7-22 16:59 | 只看该作者
你这代码就是查理复用。沙发的一次只点亮一个灯,亮度为你现在驱动的1/4,还没你现在驱动好。
消隐后要先准备数据,再拉低相应脚,使需要亮的LED同时亮。
你代码框架不好,整理了一下。

  1. #include "188.h"
  2. #include "delay.h"
  3. #include "intrins.h"
  4. #include "stc8g.h"

  5. sbit PSEG_1 = P3 ^ 6;
  6. sbit PSEG_2 = P3 ^ 5;
  7. sbit PSEG_3 = P3 ^ 4;
  8. sbit PSEG_4 = P3 ^ 3;
  9. sbit PSEG_5 = P3 ^ 2;

  10. #define PIN1_IN() \
  11.     P3M1 |= 0x40; \
  12.     P3M0 &= 0xbf; // 把单片机IO口配置为高阻输入
  13. #define PIN2_IN() \
  14.     P3M1 |= 0x20; \
  15.     P3M0 &= 0xdf;
  16. #define PIN3_IN() \
  17.     P3M1 |= 0x10; \
  18.     P3M0 &= 0xef;
  19. #define PIN4_IN() \
  20.     P3M1 |= 0x08; \
  21.     P3M0 &= 0xf7;
  22. #define PIN5_IN() \
  23.     P3M1 |= 0x04; \
  24.     P3M0 &= 0xfb;

  25. #define PIN1_L()  \
  26.     P3M0 |= 0x40; \
  27.     P3M1 &= 0xbf; \
  28.     PSEG_1 = 0; // 配置为推挽输出,且拉低
  29. #define PIN2_L()  \
  30.     P3M0 |= 0x20; \
  31.     P3M1 &= 0xdf; \
  32.     PSEG_2 = 0;
  33. #define PIN3_L()  \
  34.     P3M0 |= 0x10; \
  35.     P3M1 &= 0xef; \
  36.     PSEG_3 = 0;
  37. #define PIN4_L()  \
  38.     P3M0 |= 0x08; \
  39.     P3M1 &= 0xf7; \
  40.     PSEG_4 = 0;
  41. #define PIN5_L()  \
  42.     P3M0 |= 0x04; \
  43.     P3M1 &= 0xfb; \
  44.     PSEG_5 = 0;

  45. #define PIN1_H()  \
  46.     P3M0 |= 0x40; \
  47.     P3M1 &= 0xbf; \
  48.     PSEG_1 = 1; // 配置为推挽输出,且拉高
  49. #define PIN2_H()  \
  50.     P3M0 |= 0x20; \
  51.     P3M1 &= 0xdf; \
  52.     PSEG_2 = 1;
  53. #define PIN3_H()  \
  54.     P3M0 |= 0x10; \
  55.     P3M1 &= 0xef; \
  56.     PSEG_3 = 1;
  57. #define PIN4_H()  \
  58.     P3M0 |= 0x08; \
  59.     P3M1 &= 0xf7; \
  60.     PSEG_4 = 1;
  61. #define PIN5_H()  \
  62.     P3M0 |= 0x04; \
  63.     P3M1 &= 0xfb; \
  64.     PSEG_5 = 1;

  65. /************************************************************************************/
  66. void LED_GPIO_Init(void) // 初始化IO口
  67. {
  68.     P3PU &= 0x83; // 关闭上拉电阻
  69.     P3M1 &= 0x83; // 设置P3.2~P3.6为推挽输出
  70.     P3M0 |= 0x7c;
  71. }
  72. /********************************熄灭所有数码管***********************************/
  73. void Set_AllPin_INPUT(void)
  74. {
  75.     PIN1_IN(); // 把单片机IO口配置为输入模式
  76.     PIN2_IN();
  77.     PIN3_IN();
  78.     PIN4_IN();
  79.     PIN5_IN();
  80. }
  81. unsigned short display_sram = 0;
  82. void Display_Scan1(void)
  83. {
  84.     // PIN1_L(); // 拉低Pin1,放后面能保证4个灯同时亮
  85.     if (display_sram & 0x8000) // 取出最高位
  86.         PIN2_H();
  87.     if (display_sram & 0x4000)
  88.         PIN3_H();
  89.     if (display_sram & 0x2000)
  90.         PIN4_H();
  91.     if (display_sram & 0x1000)
  92.         PIN5_H();
  93.     PIN1_L(); // 拉低Pin1,放后面能保证4个灯同时亮
  94. }
  95. void Display_Scan2(void)
  96. {
  97.     if (display_sram & 0x0800)
  98.         PIN1_H();
  99.     if (display_sram & 0x0400)
  100.         PIN3_H();
  101.     if (display_sram & 0x0200)
  102.         PIN4_H();
  103.     if (display_sram & 0x0100)
  104.         PIN5_H();
  105.     PIN2_L(); // 拉低Pin2
  106. }
  107. void Display_Scan3(void)
  108. {
  109.     if (display_sram & 0x0080)
  110.         PIN1_H();
  111.     if (display_sram & 0x0040)
  112.         PIN2_H();
  113.     ;
  114.     if (display_sram & 0x0020)
  115.         PIN4_H();
  116.     if (display_sram & 0x0010)
  117.         PIN5_H();
  118.     PIN3_L(); // 拉低Pin3
  119. }
  120. void Display_Scan4(void)
  121. {
  122.     if (display_sram & 0x0008)
  123.         PIN1_H();
  124.     if (display_sram & 0x0004)
  125.         PIN2_H();
  126.     if (display_sram & 0x0002)
  127.         PIN3_H();
  128.     if (display_sram & 0x0001)
  129.         PIN5_H();
  130.     PIN4_L(); // 拉低Pin4
  131. }

  132. unsigned short code Segment[3][11] = {
  133.     {0x0000, 0x0006, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000}, // 百位
  134.     {0x0770, 0x0420, 0x0741, 0x0661, 0x0431, 0x0271, 0x0371, 0x0460, 0x0771, 0x0671, 0x0000}, // 十位
  135.     {0xE888, 0x8080, 0xD808, 0xD880, 0xB080, 0x7880, 0x7888, 0x8880, 0xF888, 0xF880, 0x0000}  // 个位
  136. };
  137. unsigned short g_display_sram = 0;
  138. void Dispaly_Dig(unsigned int dig) // 拆分
  139. {
  140.     unsigned char bai = dig / 100;
  141.     unsigned char shi = (dig / 10) % 10;
  142.     unsigned char ge = dig % 10;
  143.     unsigned short i16 = Segment[0][bai] | Segment[1][shi] | Segment[2][ge]; // 显示百位,十位,个位
  144.     EA = 0;
  145.     // 改变g_display_sram不是原子操作,所以要关中断。
  146.     // 增加全局变量g_display_sram,是为了尽量减少中断的运算量
  147.     g_display_sram = i16;
  148.     EA = 1;
  149. }

  150. // 定时器中断函数,5ms运行一次,扫描一轮需要4*5ms=20ms,
  151. // 也就是50Hz的刷新频率,如果感觉闪烁,可以4ms运行。
  152. void TM1_Isr() interrupt 3 // 定时器1
  153. {
  154.     // static unsigned short display_sram = 0;
  155.     static unsigned char case_cnt = 0; // 逐行扫描
  156.     Set_AllPin_INPUT();                // 消影作用
  157.     switch (case_cnt)
  158.     {
  159.     case 0x00:
  160.         Display_Scan1();
  161.         case_cnt++;
  162.         break;
  163.     case 0x01:
  164.         Display_Scan2();
  165.         case_cnt++;
  166.         break;
  167.     case 0x02:
  168.         Display_Scan3();
  169.         case_cnt++;
  170.         break;
  171.     case 0x03:
  172.         Display_Scan4();
  173.         case_cnt = 0;
  174.         display_sram = g_display_sram; // 仅在此处改变display_sram
  175.         break;
  176.     default:
  177.         case_cnt = 0;
  178.         break;
  179.     }
  180. }
复制代码


  1. /*
  2. 0~999循环显示
  3. */
  4. void Dispaly_Dig_loop()
  5. {
  6.     unsigned short i16 = 0;
  7.     while (1)
  8.     {
  9.         Dispaly_Dig(i16++);
  10.         if (i16 == 1000)
  11.             i16 = 0;
  12.         delayMs(500);
  13.     }
  14. }
复制代码

回复

使用道具 举报

9#
ID:1067504 发表于 2023-7-22 20:03 来自手机 | 只看该作者
这个数组是怎么得来的?怎么取模的?
回复

使用道具 举报

10#
ID:1073939 发表于 2023-7-24 10:24 | 只看该作者
蚂蚁科技 发表于 2023-7-22 20:03
这个数组是怎么得来的?怎么取模的?

楼主文中有链接。

回复

使用道具 举报

11#
ID:361594 发表于 2024-4-28 08:25 来自手机 | 只看该作者
我是CSDN的原创作者,评论区各位都是在瞎扯淡,这个代码已经是用在量产产品了,根本没有你们所说的问题,肯定是楼主单片机的寄存器没有配置好,
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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