JavaScript 中带前导零的数字

IT技术 javascript
2021-02-07 23:09:13

试试这个:

var num = 040;
console.log(num); // 32

从什么时候开始 40 = 32?

3个回答

使用前导零,数字被解释为八进制和4 * 8 = 32

TL; 博士

由于前导0它被视为八进制(基数 8),就像前导0x将使其变为十六进制(基数 16)一样。这有着漫长而折磨的历史,不再是现代 JavaScript 中八进制数的编写方式。在使用严格模式的现代 JavaScript 中,“传统”八进制格式是一个语法错误;八进制数写有0o前缀。

历史

早期(在 Netscape 的初始语言以及第一和第二个 ECMAScript 规范中),0数字文字的前导正式表示八进制(基数 8),就像前导0x表示十六进制(基数 16)一样:

OctalIntegerLiteral ::
    0 OctalDigit
    OctalIntegerLiteral OctalDigit

例如,10, 012, 和0xA都是十进制数十的写法。这与语法类似于 JavaScript 的其他一些语言(C、C++、Java 等)一致,但它非常令人困惑。

从 ECMAScript 3 开始,这种形式的八进制文字被降级为一个可选的扩展,十进制整数文字被更改为不能有前导零(除非实现包括扩展):

DecimalIntegerLiteral ::
    0
    NonZeroDigit DecimalDigits(opt)

但是ECMAScript 5禁止在严格模式下这样做:

在处理严格模式代码(参见 10.1.1)时符合要求的实现不得扩展NumericLiteral的语法以包含B.1.1 中所述的OctalIntegerLiteral

ECMAScript 6 (ECMAScript 2015) 引入了BinaryIntegerLiteralOctalIntegerLiteral,所以现在我们有了更连贯的文字:

  • BinaryIntegerLiteral,以0bor为前缀0B
  • OctalIntegerLiteral,以0oor为前缀0O
  • HexIntegerLiteral,以0xor为前缀0X

旧的OctalIntegerLiteral扩展已重命名为LegacyOctalIntegerLiteral,在非严格模式下仍然允许使用。

结论

因此,如果要解析基数为 8 的数字,请使用0o0O前缀(旧浏览器不支持),或使用parseInt.

如果您想确保您的数字将以 10 为基数进行解析,请删除前导零,或使用parseInt.

例子

  • 010
    • 在严格模式下(需要 ECMAScript 5),这是一个语法错误。
    • 在非严格模式下,可能是语法错误或返回8(依赖于实现)。
  • 0o10, 0O10
    • 在 ECMAScript 6 之前,它们是语法错误。
    • 在 ECMAScript 6 中,它们返回8.
  • parseInt('010', 8)
    • 它返回8
  • parseInt('010', 10)
    • 它返回10

如果你有兴趣,你可以找到目前的生活规范在这里,和历史版本在这里

因为0前缀表示一个八进制数(基数为 8)。