วิธีการปัดเศษขึ้นตัวเลขใน Javascript?


159

ฉันต้องการใช้จาวาสคริปต์เพื่อปัดเศษตัวเลข เนื่องจากตัวเลขเป็นสกุลเงินฉันต้องการให้ปัดเศษขึ้นในตัวอย่างเหล่านี้ (ทศนิยม 2 ตำแหน่ง):

  • 192.168 => 192.20
  • 192.11 => 192.20
  • 192.21 => 192.30
  • 192.26 => 192.30
  • 192.20 => 192.20

จะบรรลุสิ่งนี้ได้อย่างไรโดยใช้ Javascript ฟังก์ชัน Javascript ในตัวจะปัดเศษตัวเลขตามตรรกะมาตรฐาน (น้อยกว่าและมากกว่า 5 เพื่อปัดเศษขึ้น)

คำตอบ:


313
/**
 * @param num The number to round
 * @param precision The number of decimal places to preserve
 */
function roundUp(num, precision) {
  precision = Math.pow(10, precision)
  return Math.ceil(num * precision) / precision
}

roundUp(192.168, 1) //=> 192.2

2
@AndrewMarshall จุดประสงค์ของการคูณคืออะไรแล้วหารด้วย 10
codecowboy

6
@codecowboy ถ้าคุณทำไม่ได้แล้วceil()จะให้คุณ193ดังนั้นเราต้องให้แน่ใจว่าความแม่นยำทั้งหมดที่เราต้องการเก็บไว้ก่อนจุดทศนิยม จากนั้นเราทำการดำเนินการผกผันเพื่อคืนค่า "ดั้งเดิม"
Andrew Marshall

1
หากคุณได้รับหมายเลขเช่น192.19999999999997คุณสามารถนำ.toFixed(1)ไปใช้กับnum
flamer.ohr

4
และสำหรับคนที่นี่ที่สงสัยว่าจะปัดเศษขึ้นเป็นจำนวนที่ใกล้ที่สุดได้แค่เพียง Math.ceil () ส่วนที่เหลือเป็นเพียงการจัดการกับทศนิยม เพื่อช่วยคนอื่นให้รอดเวลาที่สมองของฉันต้องทำอย่างนั้น!
Nigel B. Peck

วิธีนี้มีข้อบกพร่อง: Math.ceil (0.0159 * 1000000000) / ความแม่นยำ คุณจะได้รับเศษส่วน 0.015900001 จำเป็นต้องเพิ่มการตรวจสอบช่วงเพื่อความแม่นยำ
แฟรงค์

26

สายน้อย แต่สามารถสร้างฟังก์ชั่นจาวาสคริปต์ที่สามารถใช้ซ้ำได้เพื่อจุดประสงค์นี้:

// Arguments: number to round, number of decimal places
function roundNumber(rnum, rlength) { 
    var newnumber = Math.round(rnum * Math.pow(10, rlength)) / Math.pow(10, rlength);
    return newnumber;
}

เรียกใช้ฟังก์ชันเป็น

alert(roundNumber(192.168,2));

2
วิธีนี้ใช้งานได้ดี แต่ OP ขอวิธีการปัดเศษตัวเลขดังนั้น Math.ceil ควรใช้ที่นี่แทน Math.round
kloddant

คำตอบนี้ดีกว่าคำตอบที่ยอมรับหากคุณต้องการปัดเศษขึ้นอย่างเหมาะสมแม้ว่าตำแหน่งทศนิยมที่คุณต้องการจะเป็น ตัวอย่าง: 1.054 -> 1.05 1.055 -> 1.06 อย่างไรก็ตามนี่คือกรณีที่ขอบ: 1.005 -> 1 1.006 -> 1.01 และ 1.015 -> 1.01 1.016 -> 1.02 ดังนั้นระวัง
Jay K

21

การปัดเศษแบบปกติจะทำงานด้วยการบิดเล็กน้อย:

Math.round(price * 10)/10

และถ้าคุณต้องการเก็บรูปแบบสกุลเงินคุณสามารถใช้วิธีการหมายเลข .toFixed()

(Math.round(price * 10)/10).toFixed(2)

แม้ว่าสิ่งนี้จะทำให้มันเป็น String =)


Math.round (192.11 * 100) / 100 -> 192.11
krtek

1
คนที่สองไม่ต้องการปัดเศษมันเป็นเหมือนprice.toFixed(2)
Michael Krelin - แฮกเกอร์

@ Krtek ooops ขอบคุณสำหรับการจับที่ ฉันอ่านคำถามผิด อัปเดตคำตอบแล้ว
Shad

2
OP ถามถึงวิธีปัดเศษตัวเลขดังนั้นควรใช้ Math.ceil ที่นี่แทน Math.round
kloddant


2

OP คาดว่าจะมีสองสิ่ง:
A. เพื่อปัดเศษเป็นสิบส่วนที่สูงขึ้นและ
B. เพื่อแสดงศูนย์ในตำแหน่งที่ร้อย (ความต้องการทั่วไปกับสกุลเงิน)

การตอบสนองความต้องการทั้งสองดูเหมือนจะจำเป็นต้องมีวิธีแยกต่างหากสำหรับแต่ละข้อ นี่คือวิธีการที่สร้างขึ้นจากคำตอบที่แนะนำของ suryakiran:

//Arguments: number to round, number of decimal places.

function roundPrice(rnum, rlength) {
    var newnumber = Math.ceil(rnum * Math.pow(10, rlength-1)) / Math.pow(10, rlength-1);
    var toTenths = newnumber.toFixed(rlength);
    return toTenths;
}

alert(roundPrice(678.91011,2)); // returns 679.00
alert(roundPrice(876.54321,2)); // returns 876.60

หมายเหตุสำคัญ: โซลูชันนี้สร้างผลลัพธ์ที่แตกต่างกันอย่างมากด้วยจำนวนลบและเลขยกกำลัง

เพื่อประโยชน์ในการเปรียบเทียบระหว่างคำตอบนี้กับสองคำที่คล้ายกันมากให้ดู 2 แนวทางต่อไปนี้ รอบแรกจะปัดเศษเป็นร้อยใกล้เคียงที่สุดต่อปกติและรอบที่สองจะปัดขึ้นเป็นร้อยใกล้เคียงที่สุด (ใหญ่กว่า)

function roundNumber(rnum, rlength) { 
    var newnumber = Math.round(rnum * Math.pow(10, rlength)) / Math.pow(10, rlength);
    return newnumber;
}

alert(roundNumber(678.91011,2)); // returns 678.91

function ceilNumber(rnum, rlength) { 
    var newnumber = Math.ceil(rnum * Math.pow(10, rlength)) / Math.pow(10, rlength);
    return newnumber;
}

alert(ceilNumber(678.91011,2)); // returns 678.92

2

ตกลงมีการตอบแล้ว แต่ฉันคิดว่าคุณอาจต้องการเห็นคำตอบของฉันที่เรียกใช้ฟังก์ชัน math.pow () หนึ่งครั้ง ฉันเดาว่าฉันชอบเก็บสิ่งที่แห้ง

function roundIt(num, precision) {
    var rounder = Math.pow(10, precision);
    return (Math.round(num * rounder) / rounder).toFixed(precision)
};

มันเป็นการรวมตัวเข้าด้วยกัน แทนที่ Math.round () ด้วย Math.ceil () เป็น round-up แทนที่จะปัดเศษออกซึ่งเป็นสิ่งที่ OP ต้องการ


1

ฟังก์ชันนี้ จำกัด ทศนิยมโดยไม่ต้องปัดเศษ

function limitDecimal(num,decimal){
     return num.toString().substring(0, num.toString().indexOf('.')) + (num.toString().substr(num.toString().indexOf('.'), decimal+1));
}

ทางเลือกที่สั้นกว่า: return ('' + num) .split ('.'). shift ()
Roberto

ขอบคุณ Roberto รหัสนี้ใช้งานได้ แต่ลบทศนิยมทั้งหมด
Behnam Mohammadi

0

ฉันใช้คำตอบ @AndrewMarshall มานาน แต่พบว่ามีบางกรณี การทดสอบต่อไปนี้ไม่ผ่าน:

equals(roundUp(9.69545, 4), 9.6955);
equals(roundUp(37.760000000000005, 4), 37.76);
equals(roundUp(5.83333333, 4), 5.8333);

นี่คือสิ่งที่ฉันใช้ตอนนี้เพื่อให้การทำงานเป็นไปอย่างถูกต้อง:

// 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

ที่มา: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Math/round


1
กรณีทดสอบของคุณดูไม่ถูกต้อง roundUp(37.760000000000005, 4)ควรจะเป็น37.7601และควรจะเป็นroundUp(5.83333333, 4) 5.8334ทั้งสอง (และของคุณก่อน) ทั้งหมดถือเป็นจริงสำหรับ fn ที่ฉันให้ไว้
Andrew Marshall

@AndrewMarshall มีเหตุผลที่ค่าคาดว่าของคุณจะไม่ถูกต้องสำหรับกรณีที่ 2 และ 3
Amn

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.