如何在 JavaScript 中用逗号打印一个数字作为千位分隔符

新手上路,请多包涵

我正在尝试用逗号作为千位分隔符在 JavaScript 中打印一个整数。例如,我想将数字 1234567 显示为“1,234,567”。我该怎么做呢?

这是我的做法:

function numberWithCommas(x) {
    x = x.toString();
    var pattern = /(-?\d+)(\d{3})/;
    while (pattern.test(x))
        x = x.replace(pattern, "$1,$2");
    return x;
}

有没有更简单或更优雅的方法来做到这一点?如果它也适用于浮点数会很好,但这不是必需的。在句点和逗号之间决定不需要特定于语言环境。

原文由 Elias Zamaria 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 1.1k
2 个回答

我使用了 Kerry 回答中的想法,但简化了它,因为我只是为我的特定目的寻找简单的东西。这是我所拥有的:

 function numberWithCommas(x) {
 return x.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ",");
 }

 function numberWithCommas(x) {
 return x.toString().replace(/\B(?<!\.\d*)(?=(\d{3})+(?!\d))/g, ",");
 }

 function test(x, expect) {
 const result = numberWithCommas(x);
 const pass = result === expect;
 console.log(`${pass ? "✓" : "ERROR ====>"} ${x} => ${result}`);
 return pass;
 }

 let failures = 0;
 failures += !test(0, "0");
 failures += !test(100, "100");
 failures += !test(1000, "1,000");
 failures += !test(10000, "10,000");
 failures += !test(100000, "100,000");
 failures += !test(1000000, "1,000,000");
 failures += !test(10000000, "10,000,000");
 if (failures) {
 console.log(`${failures} test(s) failed`);
 } else {
 console.log("All tests passed");
 }
 .as-console-wrapper {
 max-height: 100% !important;
 }

正则表达式使用 2 个前瞻断言:

  • 一个正数,用于查找字符串中后面连续有 3 个数字的倍数的任何点,

  • 一个否定断言,以确保该点仅具有 3 位数字的倍数。替换表达式在此处放置一个逗号。

例如,如果您传递 123456789.01 ,则肯定断言将匹配 7 左侧的每个点(因为 789 是 3 位数的倍数, 678 是 3 位数的倍数, 567 等)。否定断言检查 3 位的倍数后面没有任何数字。 789 后面有一个句点,所以它正好是 3 位数字的倍数,所以那里有一个逗号。 678 是 3 位数字的倍数,但它后面有一个 9 ,所以这 3 位数字是一组 4 的一部分,逗号不会去那里。对于 567 也是如此。 456789 是 6 位数字,是 3 的倍数,所以前面要加逗号。 345678 是 3 的倍数,但后面有一个 9 ,所以没有逗号。等等。 \B 防止正则表达式在字符串的开头放置逗号。

@ neu-rah 提到如果小数点后超过 3 位,此功能会在不需要的位置添加逗号。如果这是一个问题,您可以使用此功能:

 function numberWithCommas(x) {
 var parts = x.toString().split(".");
 parts[0] = parts[0].replace(/\B(?=(\d{3})+(?!\d))/g, ",");
 return parts.join(".");
 }

 function numberWithCommas(x) {
 var parts = x.toString().split(".");
 parts[0] = parts[0].replace(/\B(?=(\d{3})+(?!\d))/g, ",");
 return parts.join(".");
 }

 function test(x, expect) {
 const result = numberWithCommas(x);
 const pass = result === expect;
 console.log(`${pass ? "✓" : "ERROR ====>"} ${x} => ${result}`);
 return pass;
 }

 let failures = 0;
 failures += !test(0 , "0");
 failures += !test(0.123456 , "0.123456");
 failures += !test(100 , "100");
 failures += !test(100.123456 , "100.123456");
 failures += !test(1000 , "1,000");
 failures += !test(1000.123456 , "1,000.123456");
 failures += !test(10000 , "10,000");
 failures += !test(10000.123456 , "10,000.123456");
 failures += !test(100000 , "100,000");
 failures += !test(100000.123456 , "100,000.123456");
 failures += !test(1000000 , "1,000,000");
 failures += !test(1000000.123456 , "1,000,000.123456");
 failures += !test(10000000 , "10,000,000");
 failures += !test(10000000.123456, "10,000,000.123456");
 if (failures) {
 console.log(`${failures} test(s) failed`);
 } else {
 console.log("All tests passed");
 }
 .as-console-wrapper {
 max-height: 100% !important;
 }

@tjcrowder 指出,现在 JavaScript 有lookbehind( 支持信息),可以在正则表达式本身解决:

 function numberWithCommas(x) {
 return x.toString().replace(/\B(?<!\.\d*)(?=(\d{3})+(?!\d))/g, ",");
 }

 function numberWithCommas(x) {
 return x.toString().replace(/\B(?<!\.\d*)(?=(\d{3})+(?!\d))/g, ",");
 }

 function test(x, expect) {
 const result = numberWithCommas(x);
 const pass = result === expect;
 console.log(`${pass ? "✓" : "ERROR ====>"} ${x} => ${result}`);
 return pass;
 }

 let failures = 0;
 failures += !test(0, "0");
 failures += !test(0.123456, "0.123456");
 failures += !test(100, "100");
 failures += !test(100.123456, "100.123456");
 failures += !test(1000, "1,000");
 failures += !test(1000.123456, "1,000.123456");
 failures += !test(10000, "10,000");
 failures += !test(10000.123456, "10,000.123456");
 failures += !test(100000, "100,000");
 failures += !test(100000.123456, "100,000.123456");
 failures += !test(1000000, "1,000,000");
 failures += !test(1000000.123456, "1,000,000.123456");
 failures += !test(10000000, "10,000,000");
 failures += !test(10000000.123456, "10,000,000.123456");
 if (failures) {
 console.log(`${failures} test(s) failed`);
 } else {
 console.log("All tests passed");
 }
 .as-console-wrapper {
 max-height: 100% !important;
 }

(?<!\.\d*) 是一个否定的lookbehind,表示匹配不能以a开头 . 后跟零个或多个数字。至少在 V8 中,负后瞻比 splitjoin 解决方案( 比较)更快。

原文由 Elias Zamaria 发布,翻译遵循 CC BY-SA 4.0 许可协议

我很惊讶没有人提到 Number.prototype.toLocaleString 。它是在 JavaScript 1.5(于 1999 年推出)中实现的,因此基本上所有主流浏览器都支持它。

 var n = 34523453.345;
console.log(n.toLocaleString());    // "34,523,453.345"

从 v0.12 开始,它也可以通过包含 Intl 在 Node.js 中工作

如果您想要不同的东西, Numeral.js 可能会很有趣。

原文由 uKolka 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题