3

我正在尝试在 LCD 显示器上显示一个整数。液晶显示器的工作方式是向它发送一个 8 位 ASCII 字符并显示字符。

我到目前为止的代码是:

unsigned char text[17] = "ABCDEFGHIJKLMNOP";
int32_t n = 123456;
lcd.printInteger(text, n);

//-----------------------------------------  

void LCD::printInteger(unsigned char headLine[17], int32_t number)
{
    //......


    int8_t str[17];
    itoa(number,(char*)str,10);


    for(int i = 0; i < 16; i++)
    {
        if(str[i] == 0x0)
        break;

        this->sendCharacter(str[i]);
        _delay_ms(2);

    }
}
void LCD::sendCharacter(uint8_t character)
{
    //....

    *this->cOutputPort = character;

    //...

}

因此,如果我尝试123456在 LCD 上显示,它实际上会显示-7616,这显然不是正确的整数。

我知道可能存在问题,因为我将字符转换为有符号int8_t,然后将它们输出为无符号uint8_t。但我必须以无符号格式输出它们。我不知道如何将int32_t输入整数转换为 ASCII uint8_t-String。

4

1 回答 1

6

在您的架构上,int是一个int16_t,不是int32_t。因此,itoa视为123456-7616因为:

123456 = 0x0001_E240
 -7616 = 0xFFFF_E240

如果您将它们截断为 16 位,它们是相同的 - 这就是您的代码正在做的事情。而不是使用itoa,您有以下选项:

  1. 自己计算 ASCII 表示;
  2. 使用ltoa(long value, char * buffer, int radix)(如果可用),或
  3. 杠杆s[n]printf(如果有)。

对于最后一个选项,您可以使用以下“大部分”可移植代码:

void LCD::printInteger(unsigned char headLine[17], int32_t number) {
  ...
  char str[17];
  if (sizeof(int) == sizeof(int32_t))
    snprintf(str, sizeof(str), "%d", num);
  else if (sizeof(long int) == sizeof(int32_t))
    snprintf(str, sizeof(str), "%ld", num);
  else if (sizeof(long long int) == sizeof(int32_t))
    snprintf(str, sizeof(str), "%lld", num);
  ...
}

当且仅当您的平台没有snprintf,您可以使用sprintf和删除第二个参数 ( sizeof(str))。您的首选功能应该始终是n变体,因为它可以让您少一颗子弹来射击您的脚:)

由于您正在使用 C++ 编译器进行编译,我认为至少有一半的体面,上面应该以可移植的方式做“正确的事情”,而不会发出所有不必要的代码。传递给的测试条件if是编译时常量表达式。即使是一些相当老的 C 编译器也可以正确处理这种情况。

Nitpick:不要int8_tchar会做的地方使用。itoa, s[n]printf, 等期望char缓冲区,而不是int8_t缓冲区。

于 2016-01-13T15:48:40.607 回答