let _boundaryCheckingState = true; // 是否进行越界检查的全局开关 
 | 
  
 | 
/** 
 | 
 * 把错误的数据转正 
 | 
 * @private 
 | 
 * @example strip(0.09999999999999998)=0.1 
 | 
 */ 
 | 
function strip(num, precision = 15) { 
 | 
  return +parseFloat(Number(num).toPrecision(precision)); 
 | 
} 
 | 
  
 | 
/** 
 | 
 * Return digits length of a number 
 | 
 * @private 
 | 
 * @param {*number} num Input number 
 | 
 */ 
 | 
function digitLength(num) { 
 | 
  // Get digit length of e 
 | 
  const eSplit = num.toString().split(/[eE]/); 
 | 
  const len = (eSplit[0].split('.')[1] || '').length - +(eSplit[1] || 0); 
 | 
  return len > 0 ? len : 0; 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 把小数转成整数,如果是小数则放大成整数 
 | 
 * @private 
 | 
 * @param {*number} num 输入数 
 | 
 */ 
 | 
function float2Fixed(num) { 
 | 
  if (num.toString().indexOf('e') === -1) { 
 | 
    return Number(num.toString().replace('.', '')); 
 | 
  } 
 | 
  const dLen = digitLength(num); 
 | 
  return dLen > 0 ? strip(Number(num) * Math.pow(10, dLen)) : Number(num); 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 检测数字是否越界,如果越界给出提示 
 | 
 * @private 
 | 
 * @param {*number} num 输入数 
 | 
 */ 
 | 
function checkBoundary(num) { 
 | 
  if (_boundaryCheckingState) { 
 | 
    if (num > Number.MAX_SAFE_INTEGER || num < Number.MIN_SAFE_INTEGER) { 
 | 
      console.warn(`${num} 超出了精度限制,结果可能不正确`); 
 | 
    } 
 | 
  } 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 把递归操作扁平迭代化 
 | 
 * @param {number[]} arr 要操作的数字数组 
 | 
 * @param {function} operation 迭代操作 
 | 
 * @private 
 | 
 */ 
 | 
function iteratorOperation(arr, operation) { 
 | 
  const [num1, num2, ...others] = arr; 
 | 
  let res = operation(num1, num2); 
 | 
  
 | 
  others.forEach((num) => { 
 | 
    res = operation(res, num); 
 | 
  }); 
 | 
  
 | 
  return res; 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 高精度乘法 
 | 
 * @export 
 | 
 */ 
 | 
export function times(...nums) { 
 | 
  if (nums.length > 2) { 
 | 
    return iteratorOperation(nums, times); 
 | 
  } 
 | 
  
 | 
  const [num1, num2] = nums; 
 | 
  const num1Changed = float2Fixed(num1); 
 | 
  const num2Changed = float2Fixed(num2); 
 | 
  const baseNum = digitLength(num1) + digitLength(num2); 
 | 
  const leftValue = num1Changed * num2Changed; 
 | 
  
 | 
  checkBoundary(leftValue); 
 | 
  
 | 
  return leftValue / Math.pow(10, baseNum); 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 高精度加法 
 | 
 * @export 
 | 
 */ 
 | 
export function plus(...nums) { 
 | 
  if (nums.length > 2) { 
 | 
    return iteratorOperation(nums, plus); 
 | 
  } 
 | 
  
 | 
  const [num1, num2] = nums; 
 | 
  // 取最大的小数位 
 | 
  const baseNum = Math.pow(10, Math.max(digitLength(num1), digitLength(num2))); 
 | 
  // 把小数都转为整数然后再计算 
 | 
  return (times(num1, baseNum) + times(num2, baseNum)) / baseNum; 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 高精度减法 
 | 
 * @export 
 | 
 */ 
 | 
export function minus(...nums) { 
 | 
  if (nums.length > 2) { 
 | 
    return iteratorOperation(nums, minus); 
 | 
  } 
 | 
  
 | 
  const [num1, num2] = nums; 
 | 
  const baseNum = Math.pow(10, Math.max(digitLength(num1), digitLength(num2))); 
 | 
  return (times(num1, baseNum) - times(num2, baseNum)) / baseNum; 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 高精度除法 
 | 
 * @export 
 | 
 */ 
 | 
export function divide(...nums) { 
 | 
  if (nums.length > 2) { 
 | 
    return iteratorOperation(nums, divide); 
 | 
  } 
 | 
  
 | 
  const [num1, num2] = nums; 
 | 
  const num1Changed = float2Fixed(num1); 
 | 
  const num2Changed = float2Fixed(num2); 
 | 
  checkBoundary(num1Changed); 
 | 
  checkBoundary(num2Changed); 
 | 
  // 重要,这里必须用strip进行修正 
 | 
  return times(num1Changed / num2Changed, strip(Math.pow(10, digitLength(num2) - digitLength(num1)))); 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 四舍五入 
 | 
 * @export 
 | 
 */ 
 | 
export function round(num, ratio) { 
 | 
  const base = Math.pow(10, ratio); 
 | 
  let result = divide(Math.round(Math.abs(times(num, base))), base); 
 | 
  if (num < 0 && result !== 0) { 
 | 
    result = times(result, -1); 
 | 
  } 
 | 
  // 位数不足则补0 
 | 
  return result; 
 | 
} 
 | 
  
 | 
/** 
 | 
 * 是否进行边界检查,默认开启 
 | 
 * @param flag 标记开关,true 为开启,false 为关闭,默认为 true 
 | 
 * @export 
 | 
 */ 
 | 
export function enableBoundaryChecking(flag = true) { 
 | 
  _boundaryCheckingState = flag; 
 | 
} 
 | 
  
 | 
  
 | 
export default { 
 | 
  times, 
 | 
  plus, 
 | 
  minus, 
 | 
  divide, 
 | 
  round, 
 | 
  enableBoundaryChecking, 
 | 
}; 
 |