前端嘛,常常處理數值和時間。 因此數值和時間的格式化少不了。
最近一直在面試前端, 就出了一個如何給數值添加千分位的面試題。html
至於答案,我一直都有一種標準, 一是基於你現有的知識能夠實現, 二是超出你知識。前端
有人說, 不就一個千分位, 分分鐘解決你, 正則萬歲。 沒錯, 正則很帥, 那麼性能呢?git
我就喜歡一本正緊的瞎掰。 好了, 進入正文。github
以下有的方法,稍微改造,就能夠變成更加通用的方法,好比不是添加,而是添加#等等。面試
function format_with_array(number) { var arr = (number + '').split('.'); var int = arr[0].split(''); var fraction = arr[1] || ''; var r = ""; var len = int.length; int.reverse().forEach(function (v, i) { if (i !== 0 && i % 3 === 0) { r = v + "," + r; } else { r = v + r; } }) return r + (!!fraction ? "." + fraction : ''); }
function format_with_substring(number) { var arr = (number + '').split('.'); var int = arr[0] + ''; var fraction = arr[1] || ''; var f = int.length % 3; var r = int.substring(0, f); for (var i = 0; i < Math.floor(int.length / 3); i++) { r += ',' + int.substring(f + i * 3, f + (i + 1) * 3) } if (f === 0) { r = r.substring(1); } return r + (!!fraction ? "." + fraction : ''); }
function format_with_mod(number) { var n = number; var r = ""; do { mod = n % 1000; n = n / 1000; r = ~~mod + (!!r ? "," + r : "") } while (n > 1) var strNumber = number + ""; var index = strNumber.indexOf("."); if (index > 0) { r += strNumber.substring(index); } return r; }
function format_with_regex(number) { var reg = /\d{1,3}(?=(\d{3})+$)/g; return (number + '').replace(reg, '$&,'); } function format_with_regex(number) { var reg = /(\d)(?=(?:\d{3})+$)/g return (number + '').replace(reg, '$1,'); }
function format_with_toLocaleString(number, minimumFractionDigits, maximumFractionDigits) { minimumFractionDigits = minimumFractionDigits || 2; maximumFractionDigits = (maximumFractionDigits || 2); maximumFractionDigits = Math.max(minimumFractionDigits, maximumFractionDigits); return number.toLocaleString("en-us", { maximumFractionDigits: maximumFractionDigits || 2, minimumFractionDigits: minimumFractionDigits || 2 }) }
function format_with_Intl(number, minimumFractionDigits, maximumFractionDigits) { minimumFractionDigits = minimumFractionDigits || 2; maximumFractionDigits = (maximumFractionDigits || 2); maximumFractionDigits = Math.max(minimumFractionDigits, maximumFractionDigits); return new Intl.NumberFormat('en-us', { maximumFractionDigits: maximumFractionDigits || 2, minimumFractionDigits: minimumFractionDigits || 2 }).format(number) }
測試地址: 千分位性能測試
準備性能測試的代碼, 爲了測試帶小數位和不帶小數位,在getData的時候,若是隨機值大於0.5, 將去除小數位。chrome
function getData(count) { var data = new Array(count).fill(0).map(function (i) { var rd = Math.random(); var r = rd * Math.pow(10, 10); if (rd > 0.5) { r = ~~r; } return r }) return data; } function test(data, fn, label) { var start = performance.now(); for (var i = 0; i < data.length; i++) { fn(data[i]); } var time = performance.now() - start; message((fn.name || label) + ":" + time + "ms"); } function executeTest() { var data = getData(+textCount.value); test(data, format_with_array); test(data, format_with_mod); test(data, format_with_substring); test(data, format_with_regex); test(data, format_with_toLocaleString); test(data, format_with_Intl); } function message(msg) { var el = document.createElement("p"); el.innerHTML = msg; messageEl.appendChild(el); }
測試數據50000瀏覽器
format_with_array:59.13ms緩存
format_with_mod:23.96msapp
format_with_substring:44.04msdom
format_with_regex:53.54ms
format_with_toLocaleString:1813.61ms
format_with_Intl:1973.45ms
format_with_array:63.30ms
format_with_mod:37.80ms
format_with_substring:41.40ms
format_with_regex:51.20ms
format_with_toLocaleString:3334.30ms
format_with_Intl:3485.80ms
兼容模式不支持 Array.fill Pass
format_with_array:75.29ms
format_with_mod:35.47ms
format_with_substring:40.79ms
format_with_regex:49.86ms
format_with_toLocaleString:2418.04ms
format_with_Intl:2474.30ms
format_with_array:41.00ms
format_with_mod:25.00ms
format_with_substring:28.00ms
format_with_regex:43.00ms
format_with_toLocaleString:1799.00ms
format_with_Intl:2239.00ms
基本的結果都是一致的, 由於隨機數不同,結果也會偶爾不一致。
format_with_mod > format_with_substring > format_with_regex > format_with_array > format_with_toLocaleString > format_with_Intl
其中Intl.NumberFormat是有很大提高空間的,能夠把實例緩存之類的。