如果前导0后面的数字是8或9,Scanf不会读取前导0

问题描述

我有一个非常奇怪的问题,源于我一直在编写的C ++ 11程序的错误

请参见以下代码

long long a[1000];
int main(int argc,char * argv[]) {
    
    for(long long i = 0; i < 300; ++i) {
        scanf("%lli",&a[i]);
        std::cout << a[i] << std::endl;
    }
    
    return 0;
}

尝试输入12等,我们得到预期的输出1\n2\n等。这同样适用于输入0011\n00044\n的输入。

但是,当前导零之后的数字是8或9时,scanf()首先读取前导零,然后再读取数字。

例如:

输入:0009输出000\n9\n

输入:08输出0\n8\n

输入:00914输出00\n914\n

我已经进行了一些测试,在这些情况下,scanf()似乎首先读取了前导零,而其余的数字则留在缓冲区中,并在循环的第二次运行中被拾取。 / p>

有人可以暗示发生了什么事吗?

我正在使用XCode 11.3.7并使用Clang C ++ 11进行编译。 (我没有弄乱项目设置)

提前谢谢!!

解决方法

使用%lld代替%lli

%i不起作用的原因是因为0interpreted作为八进制数字的前缀,而数字89却不t以八进制形式存在:

d匹配一个可选的带符号十进制整数;下一个指针必须是指向int的指针。

i匹配一个可选的带符号整数;下一个指针必须是指向int的指针。如果整数以0x或0X开头,则以16为基数;如果以0开头,则以8为基数;否则以10为基数。仅字符 使用与基础相对应的。

对于其他数字,您也会得到错误的答案,例如八进制的010将被解析为8。

或者,甚至更好:使用C ++代替C。

std::cin >> a[i];