找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索

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

查看数: 6947 | 评论数: 14 | 收藏 6
关灯 | 提示:支持键盘翻页<-左 右->
    组图打开中,请稍候......
发布时间: 2022-2-12 15:03

正文摘要:

大家好,请教一下,我在使用188数码管时,遇到了一个消影问题,数码管动态扫描,在刷新函数中,已经加入了消影语句,即各个引脚设置为高阻输入,但是不应该亮的数码管还是会有弱光,所以显示并不清晰,当然消影函数 ...

回复

ID:828160 发表于 2024-7-18 20:46
这么简单的数码管,直接用带LED串行点阵的单片机就可以驱动了,最多9个IO驱动64个LED,而且还能调亮度,不仅能调16级亮度,还可以让同时亮的LED显示两种不同的亮度。

BF6612SCXXB-XJLK-MCU CN-Datasheet-V1.0-22-03-03.pdf (5.52 MB, 下载次数: 13)

ID:59202 发表于 2024-7-18 19:30
P3M1/P3M0是STC单片机的P3口配置寄存器,拿来设置I/O口工作模式的
ID:619259 发表于 2024-7-18 12:51
关于188数码管,经过研究与实践,几点总结:
1.单片机IO口要有推挽输出和高阻输入模式,我用的是STC32G单片机,在实践中采用推挽输出和高阻输入两种模式配合驱动。(使用准双向和高阻模式时,亮度暗,不推荐)。
2.先测试各引脚的电平与字段的对应关系,得出引脚驱动电平关系表(红色为高电平、蓝色为低电平)。

3.根据驱动电平关系表配置IO口。
下图为IO配置代码表,本案例使用P1.3-P1.7驱动。

4.逐段驱动(总20段),动态扫描。
以第三位为例,扫描程序如下:

每扫描完成一个字段,都需要消隐,不然会出现鬼影。
显示效果如下图:

ID:1125252 发表于 2024-7-18 08:43
问下P3M1/P3M0分别指代什么?是分别指代共阳端和负极吗?
ID:361594 发表于 2024-4-28 08:25
我是CSDN的原创作者,评论区各位都是在瞎扯淡,这个代码已经是用在量产产品了,根本没有你们所说的问题,肯定是楼主单片机的寄存器没有配置好,
ID:1073939 发表于 2023-7-24 10:24
蚂蚁科技 发表于 2023-7-22 20:03
这个数组是怎么得来的?怎么取模的?

楼主文中有链接。

ID:1067504 发表于 2023-7-22 20:03
这个数组是怎么得来的?怎么取模的?
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. }
复制代码

ID:1067504 发表于 2023-7-21 22:00
能不能分享完整的源代码,我想学习一下,谢谢!
ID:399179 发表于 2022-3-26 21:58
哎呀!我也遇到这样的问题了
ID:575020 发表于 2022-2-13 23:40
以前还在纠结数码管用573、595、还有TM、MAX系列到底那种方式好,今天看到这种5脚2位半数码管又彻底傻眼了,想不通怎么会有如此怪异设计的数码管。

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

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

谢谢学习一下
ID:313048 发表于 2022-2-12 16:22
先显示再消影,你先消影再显示就相当于程序跑了一圈,然后你开头灭一下 大部分时间都是亮的
ID:213173 发表于 2022-2-12 15:03
采用查理复用IO驱动方式,扫描周期小于1.25ms,每次只需点亮一个笔段,分16次完成一轮,不必消隐。 查理复用IO驱动数码管.docx (92.31 KB, 下载次数: 166)

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

Powered by 单片机教程网

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