如何在javascript中舍入浮点数?


298

我需要一轮例如6.6886896.7,但它总是显示我7

我的方法:

Math.round(6.688689);
//or
Math.round(6.688689, 1);
//or 
Math.round(6.688689, 2);

但是结果总是一样的7……我在做什么错?


28
(6.688689​).toFixed(1);
暗影巫师为您耳边


JavaScript中的整数可能重复到小数点后N位 -在提出新问题之前,请先使用搜索功能。
Felix Kling


1
@ShadowWizard是正确的。但是,.toFixed返回一个字符串。确保将结果包装在中Number()。参见接受的答案和其他。
乔丹·阿塞诺

Answers:


534
Number((6.688689).toFixed(1)); // 6.7

22
将数字转换为字符串,然后再次返回?那不可能很快。
csl 2012年

如果您的数字小数位数少于所需位数,则不好。它添加了一些
njzk2

2
如JS基准测试所示,它比@fivedigit方法要慢
fadomire

13
Number((456.1235).toFixed(3)) -> 456.123Number((1.235).toFixed(2)) -> 1.24... Stupid JavaSript ...
NoOne 2015年

6
这可能无法满足您的期望!结果甚至可能取决于浏览器,请参见以下问题:stackoverflow.com/q/566564/2224996
maja 2015年

199
var number = 6.688689;
var roundedNumber = Math.round(number * 10) / 10;

13
这并不总是有效。就拿Math.round(1.005*100)/100例如,从MDN
tybro0103

7
@ tybro0103浮点数邪恶:(1.005 * 100 = 100.49999999999999至少在我尝试过的JS引擎中)。这就是为什么它不起作用以及为什么您永远不应该依赖浮点数精确的原因。
sudo

1
错误。Math.round(1.015 * 100)/ 100返回1.01,而不是1.02
Marco Marsala

81

使用toFixed()功能。

(6.688689).toFixed(); // equal to 7
(6.688689).toFixed(1); // equal to 6.7
(6.688689).toFixed(2); // equal to 6.69

5
这可能无法满足您的期望!结果甚至可能取决于浏览器,请参见以下问题:stackoverflow.com/q/566564/2224996
maja 2015年

9
(6.688689).toFixed(); 等于“ 7”而不是7。其他示例相同。
瓦多2015年

35

更新(2019-10)。多亏了Reece Daniels的以下代码,现在可以作为一组功能封装在npm-package 预期轮次中(请看一下)。


您可以使用MDN示例中的辅助函数。比您拥有更大的灵活性:

Math.round10(5.25, 0);  // 5
Math.round10(5.25, -1); // 5.3
Math.round10(5.25, -2); // 5.25
Math.round10(5, 0);     // 5
Math.round10(5, -1);    // 5
Math.round10(5, -2);    // 5

更新(2019-01-15)。似乎MDN文档不再具有此辅助功能。这是带有示例的备份:

// Closure
(function() {
  /**
   * Decimal adjustment of a number.
   *
   * @param {String}  type  The type of adjustment.
   * @param {Number}  value The number.
   * @param {Integer} exp   The exponent (the 10 logarithm of the adjustment base).
   * @returns {Number} The adjusted value.
   */
  function decimalAdjust(type, value, exp) {
    // If the exp is undefined or zero...
    if (typeof exp === 'undefined' || +exp === 0) {
      return Math[type](value);
    }
    value = +value;
    exp = +exp;
    // If the value is not a number or the exp is not an integer...
    if (isNaN(value) || !(typeof exp === 'number' && exp % 1 === 0)) {
      return NaN;
    }
    // If the value is negative...
    if (value < 0) {
      return -decimalAdjust(type, -value, exp);
    }
    // Shift
    value = value.toString().split('e');
    value = Math[type](+(value[0] + 'e' + (value[1] ? (+value[1] - exp) : -exp)));
    // Shift back
    value = value.toString().split('e');
    return +(value[0] + 'e' + (value[1] ? (+value[1] + exp) : exp));
  }

  // Decimal round
  if (!Math.round10) {
    Math.round10 = function(value, exp) {
      return decimalAdjust('round', value, exp);
    };
  }
  // Decimal floor
  if (!Math.floor10) {
    Math.floor10 = function(value, exp) {
      return decimalAdjust('floor', value, exp);
    };
  }
  // Decimal ceil
  if (!Math.ceil10) {
    Math.ceil10 = function(value, exp) {
      return decimalAdjust('ceil', value, exp);
    };
  }
})();

用法示例:

// Round
Math.round10(55.55, -1);   // 55.6
Math.round10(55.549, -1);  // 55.5
Math.round10(55, 1);       // 60
Math.round10(54.9, 1);     // 50
Math.round10(-55.55, -1);  // -55.5
Math.round10(-55.551, -1); // -55.6
Math.round10(-55, 1);      // -50
Math.round10(-55.1, 1);    // -60
Math.round10(1.005, -2);   // 1.01 -- compare this with Math.round(1.005*100)/100 above
Math.round10(-1.005, -2);  // -1.01
// Floor
Math.floor10(55.59, -1);   // 55.5
Math.floor10(59, 1);       // 50
Math.floor10(-55.51, -1);  // -55.6
Math.floor10(-51, 1);      // -60
// Ceil
Math.ceil10(55.51, -1);    // 55.6
Math.ceil10(51, 1);        // 60
Math.ceil10(-55.59, -1);   // -55.5
Math.ceil10(-59, 1);       // -50

2
?? “ Math.round10不是函数”
彼得·克劳斯

1
我发现这非常有用,因此将其打包在一个快速的npm软件包中,以供那些希望使用它而不会膨胀额外代码的人使用。最初的答案也归功于@aspanchenko:npmjs.com/package/expected-round
Reece Daniels

19
> +(6.688687).toPrecision(2)
6.7

一个Number在JavaScript对象具有不正是你需要的方法。那方法是Number.toPrecision([precision])

就像.toFixed(1)将结果转换为字符串一样,需要将其转换回数字。在+此处使用前缀完成。

我的笔记本电脑上的简单基准测试:

number = 25.645234 typeof number
50000000 x number.toFixed(1) = 25.6 typeof string / 17527ms
50000000 x +(number.toFixed(1)) = 25.6 typeof number / 23764ms
50000000 x number.toPrecision(3) = 25.6 typeof string / 10100ms
50000000 x +(number.toPrecision(3)) = 25.6 typeof number / 18492ms
50000000 x Math.round(number*10)/10 = 25.6 typeof number / 58ms
string = 25.645234 typeof string
50000000 x Math.round(string*10)/10 = 25.6 typeof number / 7109ms

我认为这是最好的答案(就最佳实践而言最好,最直接的方法)。
cezar

1
1e-10.toPrecision(3):“ 1.00e-10”-这四舍五入为有效数字
Vsevolod Golovanov

9

如果你不仅想用toFixed(),但也ceil()floor()上浮动,那么你可以使用下面的功能:

function roundUsing(func, number, prec) {
    var tempnumber = number * Math.pow(10, prec);
    tempnumber = func(tempnumber);
    return tempnumber / Math.pow(10, prec);
}

产生:

> roundUsing(Math.floor, 0.99999999, 3)
0.999
> roundUsing(Math.ceil, 0.1111111, 3)
0.112

UPD:

另一种可能的方式是这样的:

Number.prototype.roundUsing = function(func, prec){
    var temp = this * Math.pow(10, prec)
    temp = func(temp);
    return temp / Math.pow(10, prec)
}

产生:

> 6.688689.roundUsing(Math.ceil, 1)
6.7
> 6.688689.roundUsing(Math.round, 1)
6.7
> 6.688689.roundUsing(Math.floor, 1)
6.6

然后,我们如何确定何时使用ceil,floor或round进行舍入?例如:roundUsing(Math.round,1.015,2)roundUsing(Math.ceil,1.015,2)给出2个不同的值我们如何知道要使用哪个值
gaurav5430 19/12/26

7

我的扩展回合功能:

function round(value, precision) {
  if (Number.isInteger(precision)) {
    var shift = Math.pow(10, precision);
    // Limited preventing decimal issue
    return (Math.round( value * shift + 0.00000000000001 ) / shift);
  } else {
    return Math.round(value);
  }
} 

示例输出:

round(123.688689)     // 123
round(123.688689, 0)  // 123
round(123.688689, 1)  // 123.7
round(123.688689, 2)  // 123.69
round(123.688689, -2) // 100
round(1.015, 2) // 1.02

1
这类似于@fivedigit的答案。round(
1.015,2

@ gaurav5430,感谢您的举报,我们已修复此问题。
蔡仔

6

见下文

var original = 28.59;

var result=Math.round(original*10)/10 会回来的 28.6

希望这就是你想要的。


7
“如果每次看到有人使用FLOAT来存储货币时,我都会得到一角钱,那么我将有999.997634美元。”-Bill Karwin。
emix

2
错误。Math.round(1.015 * 100)/ 100
Marco Marsala

3
float(value,ndec);
function float(num,x){
this.num=num;
this.x=x;
var p=Math.pow(10,this.x);
return (Math.round((this.num).toFixed(this.x)*p))/p;
}

2

我认为此功能可以提供帮助。

 function round(value, ndec){
    var n = 10;
    for(var i = 1; i < ndec; i++){
        n *=10;
    }

    if(!ndec || ndec <= 0)
        return Math.round(value);
    else
        return Math.round(value * n) / n;
}


round(2.245, 2) //2.25
round(2.245, 0) //2

与其他一些答案类似,这在round(
1.015,2

1

我认为下面的功能可以帮助

function roundOff(value,round) {
   return (parseInt(value * (10 ** (round + 1))) - parseInt(value * (10 ** round)) * 10) > 4 ? (((parseFloat(parseInt((value + parseFloat(1 / (10 ** round))) * (10 ** round))))) / (10 ** round)) : (parseFloat(parseInt(value * (10 ** round))) / ( 10 ** round));
}

用法:roundOff(600.23458,2);将返回600.23


您能解释一下此答案所添加的内容吗?
steeththeninja

与其他一些答案类似,这在round(
1.015,2

1

如果您在node.js上下文中,则可以尝试mathjs

const math = require('mathjs')
math.round(3.1415926, 2) 
// result: 3.14

1

还有一个.toLocaleString()来格式化数字,其中有很多关于语言环境,分组,货币格式和符号的选项。一些例子:

四舍五入到小数点后一位,返回浮点数:

const n = +6.688689.toLocaleString('fullwide', {maximumFractionDigits:1})
console.log(
  n, typeof n
)

四舍五入到小数点后2位,格式为带有指定符号的货币,对千位使用逗号分组:

console.log(
  68766.688689.toLocaleString('fullwide', {maximumFractionDigits:2, style:'currency', currency:'USD', useGrouping:true})   
)

格式为区域货币:

console.log(
  68766.688689.toLocaleString('fr-FR', {maximumFractionDigits:2, style:'currency', currency:'EUR'})   
)

四舍五入到最小3位小数,强制显示零:

console.log(
  6.000000.toLocaleString('fullwide', {minimumFractionDigits:3})
)

比率的百分比样式。输入* 100,带%符号

console.log(
  6.688689.toLocaleString('fullwide', {maximumFractionDigits:2, style:'percent'})
)


1
+((6.688689 * (1 + Number.EPSILON)).toFixed(1)); // 6.7
+((456.1235 * (1 + Number.EPSILON)).toFixed(3)); // 456.124

1

如果toFixed()无法正常工作,我有很好的解决方案。

function roundOff(value, decimals) {
  return Number(Math.round(value+'e'+decimals)+'e-'+decimals);
}

roundOff(10.456,2) //output 10.46


0

这个答案进行细微调整:

function roundToStep(value, stepParam) {
   var step = stepParam || 1.0;
   var inv = 1.0 / step;
   return Math.round(value * inv) / inv;
}

roundToStep(2.55, 0.1) = 2.6
roundToStep(2.55, 0.01) = 2.55
roundToStep(2, 0.01) = 2

roundToStep(1.015,0.002)得到1.014。是使用它的正确方法
gaurav5430 '19

@ gaurav5430试试roundToStep(1.015, 0.001)
Crashalot
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.