scMathUtil.js 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. /**
  2. * Created by jimiz on 2017/3/28.
  3. * 经验证:10000次四舍五入,用num.toFixed为15毫秒,用roundTo为47毫秒,速度低一些,但可以接受
  4. * 另:经手工验证,用num.toString(2)将十进制浮点数转换为二进制浮点数时,最后一位有错误的情况出现,例子(10.0311)
  5. */
  6. let scMathUtil = {
  7. innerRoundTo: function(num, digit){
  8. let lFactor = Math.pow(10, digit);
  9. let fOffSet = 0.5;
  10. let sign = '';
  11. // 处理符号
  12. if (num < 0){
  13. sign = '-';
  14. num = Math.abs(num);
  15. }
  16. // 计算
  17. let result = Math.floor((num / lFactor) + fOffSet).toString();
  18. let iLength = result.length;
  19. // 因为数值被转为整数计算,当目标位数在小数点后,且数值小于0,需要补齐前面的位数
  20. if (iLength < -digit){
  21. result = this.zeroString(-digit) + result;
  22. }
  23. // 当目标位数在小数点前,需要补齐后面的位数
  24. else if ((digit > 0) && (iLength < digit)){
  25. result = result + this.zeroString(digit);
  26. }
  27. iLength = result.length;
  28. // 获得小数点前的数字
  29. let r1 = result.substring(0, iLength + digit);
  30. // 获得小数点后的数字
  31. let r2 = result.substring(iLength + digit, iLength);
  32. // 拼出完整结果
  33. return Number(sign + r1 + '.' + r2);
  34. },
  35. // 原来直接用num.toString(2),如果小数段最后位数是0,会被舍掉,导致进位计算bug
  36. // 改为自己计算二进制,固定为53位。
  37. // 经验证速度没有差别
  38. // 另:经手工验证,用num.toString(2)将十进制浮点数转换为二进制浮点数时,最后一位有错误的情况出现,例子(10.0311)
  39. floatToBin: function(num) {
  40. let sign = '';
  41. let dNum = num;
  42. // 符号位
  43. if (num < 0) {
  44. sign = '-';
  45. dNum = -num;
  46. };
  47. // 解析整数段
  48. let iNum = Math.floor(dNum);
  49. let iFactor;
  50. let sResult1 = '';
  51. // 计算二进制整数段
  52. while (iNum > 0){
  53. iFactor = iNum % 2;
  54. iNum = Math.floor(iNum / 2);
  55. sResult1 = iFactor + sResult1;
  56. }
  57. // 判断是否有整数段
  58. let bIntZero = sResult1 === '';
  59. if (bIntZero){
  60. sResult1 = '0';
  61. }
  62. // 解析小数段
  63. let fNum = dNum - Math.floor(dNum);
  64. let sResult2 = '';
  65. if (fNum > 0){
  66. // 双精度浮点数,尾数总长52位,因为第一位总是1,存储时已经被隐藏,所以有效位数为53位
  67. const floatLength = 53;
  68. let iLength;
  69. // js的bug,浮点数直接取小数可能不能获得精确值,只有转成字符串,截取字符串中的小数段
  70. let sNum = dNum.toString(10);
  71. let iDot = sNum.indexOf('.');
  72. sNum = '0' + sNum.substring(iDot, sNum.length);
  73. fNum = Number(sNum);
  74. // 有整数段,则小数位数为全部位数-整数位数
  75. if (!bIntZero){
  76. iLength = floatLength - sResult1.length;
  77. }
  78. else{
  79. iLength = floatLength;
  80. }
  81. // 计算二进制小数段
  82. while (iLength > 0){
  83. fNum = fNum * 2;
  84. iFactor = Math.floor(fNum);
  85. fNum = fNum % 1;
  86. sResult2 = sResult2 + iFactor;
  87. if (iFactor > 0){
  88. bIntZero = false;
  89. }
  90. if (bIntZero && (iFactor === 0)){
  91. continue;
  92. }
  93. iLength--;
  94. }
  95. }
  96. return sign + sResult1 + '.' + sResult2;
  97. },
  98. binToFloat: function(bin) {
  99. let result = 0;
  100. let iLength = bin.length;
  101. let sign = '';
  102. if (iLength > 0 && bin[0]==='-'){
  103. sign = '-';
  104. bin = bin.substring(1, iLength);
  105. }
  106. iLength = bin.length;
  107. let iDot = bin.indexOf('.');
  108. if (iDot >= 0) {
  109. for (let i = 0; i < iLength; i++) {
  110. let num = Number(bin[i]);
  111. let idx = iDot - i;
  112. if (idx === 0) {
  113. continue
  114. };
  115. if (idx > 0) {
  116. idx -= 1
  117. };
  118. let r = Math.pow(2, idx);
  119. result += num * r;
  120. }
  121. }
  122. else {
  123. result = parseInt(bin, 2);
  124. };
  125. return sign + result;
  126. },
  127. zeroString: function(length){
  128. let result = '';
  129. for (let i = 0; i < length; i++){
  130. result = result + '0';
  131. };
  132. return result;
  133. },
  134. incMantissa: function(bin){
  135. let result = bin;
  136. let iDot = bin.indexOf('.');
  137. if (iDot < 0){return result};
  138. let iLength = bin.length;
  139. iLength = bin.length;
  140. for (let i = iLength - 1; i > iDot; i--){
  141. let num = Number(bin[i]);
  142. if (num === 0){
  143. num = 1;
  144. let bin1 = bin.substring(0, i);
  145. let bin2 = this.zeroString(iLength - (i + 1));//bin.substring(i + 1, iLength);
  146. result = bin1 + num.toString() + bin2;
  147. break;
  148. }
  149. };
  150. return result;
  151. },
  152. roundTo: function(num, digit){
  153. let me = this;
  154. return me.innerRoundTo(me.binToFloat(me.incMantissa(me.floatToBin(num))), digit);
  155. },
  156. isNumber : function (obj) {
  157. return obj === +obj;
  158. },
  159. roundToString:function(obj,decimal){
  160. let me = this;
  161. let value;
  162. if(me.isNumber(obj)){
  163. value = me.roundTo(obj,-decimal)
  164. }else {
  165. value = me.roundTo(Number(obj),-decimal);
  166. }
  167. return value.toFixed(decimal);
  168. }
  169. };
  170. Number.prototype.toDecimal = function (ADigit) {
  171. //return parseFloat(this.toFixed(ADigit));
  172. digit = (ADigit && typeof(ADigit) === 'number' && Number.isInteger(ADigit) && ADigit >= 0) ? -ADigit : -2;
  173. // var s = scMathUtil.roundTo(this, digit);
  174. // console.log('Number: ' + this + ' Digit: ' + digit + ' Result: ' + s);
  175. // return parseFloat(s);
  176. return scMathUtil.roundTo(this, digit);
  177. };