MPLAB XC8 编译器 PIC18F452 多路复用七段显示代码工作正常

MPLAB XC8 Compiler PIC18F452 Multiplexed Seven Segment Display Code working properly

我目前正在编写涉及 MPLAB XC8 编译器 PIC18F452 和多路复用七段显示器的代码。我想使用连接到 PIC18F452 PORTB 引脚 RB2 和 RB3 的两个按钮来递增和递减变量 "count" 并在此显示器上显示从 0 到 99 的数字。原理图和代码如下所示。

这段代码相对正常,我不认为原理图是我所看到的问题的罪魁祸首,字节数组也不正确,因为在使用数组时我能够看到每个数字1 段显示。

尝试使用下图所示的多路复用方案时出现问题。我可以在七段显示器上成功显示两个数字,但是执行这段代码时出现奇怪的异常情况。首先,我似乎无法在任何一个显示器上显示数字 1、4 和偶尔显示 7,但是当这个数字不显示时,显示器是空白的,当再次按下按钮时,下一个数字会按预期显示。

例如:

显示屏显示数字规则如下: 9...10...11...12 13...14...等等...

34..... 35...36...37.....

不确定问题出在哪里,调试也不顺利...任何帮助将不胜感激。

Schematic for Multiplexed 7 Segment Display

#define _XTAL_FREQ 10000000
#include <xc.h>
#include <stdlib.h>
#define Digit1 PORTBbits.RB1 //variable to sink current to PNP base
#define Digit2 PORTBbits.RB2 //variable to sink current to PNP base
#define Switch1 PORTBbits.RB4 //switch decrement variable
#define Switch2 PORTBbits.RB3 //switch increment variable
#define Pressed1 1 //pressed is high
#define Pressed2 1 //pressed is high
void initialize();
void segment1 (void);
void segment2 (void);
void buttonPress(void);
void delay_ms(unsigned int);
void sendPattern1(unsigned char pattern);
void sendPattern2(unsigned char pattern3);
unsigned char rotateLeft(unsigned char pattern, int no);
unsigned char MSD, LSD, count=0;

主要代码

void main(void){
  initialize();
  Digit1 = 0;
  Digit2 = 0;
  while(1){
  buttonPress();
  MSD = count/10 ;  
  segment1();
  Digit2 = 1;
  delay_ms(10); // Delay for 10 ms
  Digit2 = 0;
  LSD = count%10; 
  segment2();
  Digit1 = 1;
  delay_ms(10); // Delay for 10 ms
  Digit1 = 0;
   }
    return;
}

索引数组中最高有效数字和最低有效数字的函数,这些数字将被发送到端口以吸收低电流以用于公共阳极显示。

void segment1(void){
unsigned char segArrayC[]={0b11000000,0b11111001,0b00100100, 
0b00110000,0b00011001,0b00010010, 
0b00000010,0b11111000,0b00000000,0b00011000};
 unsigned char pattern;
    pattern = segArrayC[MSD];
    sendPattern1(pattern);
    return;
}
void segment2(void){
unsigned char segArrayD[]= {0b11000000,0b11111001,0b00100100,
0b00110000,0b00011001,0b00010010,0b00000010,
0b11111000,0b00000000,0b00011000};
unsigned char pattern3;
     pattern3 = segArrayD[LSD];
     sendPattern2(pattern3);
     return;
}

按钮按下代码

void buttonPress(void){
  if (Switch1 == Pressed1) {
        ++count;
        delay_ms(100);
    }
 if (Switch2 == Pressed2) {
        --count;
        delay_ms(100);
    }

 if(count>=99||count<0)
 {
     count=0; 
     delay_ms(100);
 }
  return;
}

将数组中的字节向左旋转两个位置以显示在 PORT 上的功能

/** Rotate pattern to the left 'no' number of bits
 */
unsigned char rotateLeft(unsigned char pattern, int no) {
    return (((pattern << no) & 0xFF) | (pattern >> (8-no)));
}

将索引数组字符输出到 PORTC 和 PORTB 引脚的函数

void sendPattern1(unsigned char pattern) {
    // Send pattern to appropriate port pins
    unsigned char pattern2;
    PORTC = pattern;
    pattern2=rotateLeft(pattern, 2);
    PORTB = pattern2;
    return;
}
void sendPattern2(unsigned char pattern3) {
    unsigned char pattern4;
    PORTC = pattern3;
    pattern4=rotateLeft(pattern3, 2);
    PORTB = pattern4;
    return;
}

延时功能

    void delay_ms(unsigned int n){
    while (--n) _delay(2500);
}

初始化要使用的引脚(0 输出,1 输入)

void initialize() {
     TRISC = 0;
     TRISBbits.TRISB0 = 0;
     TRISBbits.TRISB1 = 0;
     TRISBbits.TRISB2 = 0;
     TRISBbits.TRISB4 = 1;
     TRISBbits.TRISB3 = 1;
     PORTC = 0x00;
     PORTB = 0x00;
}

在 sendPattern() 中,您将旋转位模式写入 PORTB。 这会干扰共阳极控制的设置。所以如果两个右手段都打开,您只能看到两个数字。根据你的示意图,你应该写一个 0 来打开共阳极。试试这个:

void main()
{
     static const unsigned char segArray[]= 
     { 0b11000000, 0b11111001, 0b00100100, 0b00110000, 0b00011001,
       0b00010010, 0b00000010, 0b11111000, 0b00000000, 0b00011000
     };
     TRISC = 0; //PortC all OUTPUT
     PORTC = 0xFF; //PortC all HIGH = IDLE = LED_OFF

     TRISBbits.TRISB0 = 0; //Output unused
     TRISBbits.TRISB1 = 0; //Output Digit1
     TRISBbits.TRISB2 = 0; //Output Digit2
     TRISBbits.TRISB4 = 1; //Input: Switch PLUS
     TRISBbits.TRISB3 = 1; //Input: Switch MINUS 

     PORTB = 0x00;
     unsigned char count=0;
     for(;;)
     {
        //Handle buttons
        if (Switch1 && count<99) 
        {
           ++count;
           delay_ms(100);
        }
        if (Switch2 && count > 0) 
        {
           --count;
           delay_ms(100);
        }

        //Write high digit
        PORTC = segArray[count/10];  
        Digit2 = 0;
        delay_ms(10); // Delay for 10 ms
        Digit2 = 1;

        //Write low digit
        PORTC = segArray[count%10];  
        Digit1 = 0;
        delay_ms(10); // Delay for 10 ms
        Digit1 = 1;
   }
}