数值千分位,6种方法和性能对比,值得一看

简介: 前端嘛,经常处理数值和时间。 所以数值和时间的格式化少不了。最近一直在面试前端, 就出了一个如何给数值添加千分位的面试题。至于答案,, 一是基于你现有的知识实现, 二是知识面。有人说, 不就一个千分位, 分分钟解决你, 正则万岁。 没错, 正则很帅, 那么性能呢? 好了, 进入正文。

1.JPG


前提



前端嘛,经常处理数值和时间。 所以数值和时间的格式化少不了。


最近一直在面试前端, 就出了一个如何给数值添加千分位的面试题。

至于答案,, 一是基于你现有的知识实现, 二是知识面。 有人说, 不就一个千分位, 分分钟解决你, 正则万岁。 没错, 正则很帅, 那么性能呢?


我就喜欢一本正紧的瞎掰。 好了, 进入正文。


实现



如下有的方法,稍微改造,就可以变成更加通用的方法,比如不是添加,而是添加#等等。


数值转字符串遍历


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 : '');
}
复制代码


substring


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 = ""; 
    var temp;
    do {
        mod = n % 1000;
        n = n / 1000;
        temp = ~~mod;
        r =  (n >= 1 ?`${temp}`.padStart(3, "0"): temp) + (!!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,');
}
复制代码


toLocaleString


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
    })
}
复制代码


Intl.NumberFormat


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, 将去除小数位。


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


1. chrome 版本 74.0.3729.131(正式版本) (32 位)


format_with_array:59.13ms

format_with_mod:23.96ms

format_with_substring:44.04ms

format_with_regex:53.54ms

format_with_toLocaleString:1813.61ms

format_with_Intl:1973.45ms


2. 360极速浏览器11.0.2052.0


极速模式


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


3. 搜狗高速浏览器 8.5.10.30358


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


4. firefox 66.0.3


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是有很大提升空间的,可以把实例缓存之类的。


当然,测试可能不严谨,问题不大,知道有这么几种思路就行,还有哪些思路呢? 欢迎留言。



相关文章
|
8月前
|
算法
算法编程(二十九):统计一致字符串的数目
算法编程(二十九):统计一致字符串的数目
96 0
|
存储 算法
算法小白的心得笔记:比较小数点后五位,而不会受到浮点数精度问题的影响。
std::cerr << "\n __" << inum << "__ 计算错误 " << ratio << " 应该是 " << beta3[inum - 1] << std::endl; return 1;
52 0
代码随想录Day22 LeetCode T39 组合总和 T40 组合总和II T131 分割回文串
代码随想录Day22 LeetCode T39 组合总和 T40 组合总和II T131 分割回文串
39 0
学C的第十三天【应用多文件的形式实现 三子棋 程序(重点);练习:1. 打印9*9乘法口诀表、2. 求10个整数中的最大值、3. 分数加减交叉计算、4. 数一下 1到 100 的整数中出现了多少个9】
9.数组的应用实例1:三子棋(综合以前学习的知识) 三子棋的实现:(重点都在注释中) 1. 游戏不退出,继续玩下一把(循环) 2. 应用多文件的形式写代码
|
算法 Java 网络架构
代码随想录训练营day27| 39. 组合总和 40.组合总和II 131.分割回文串
代码随想录训练营day27| 39. 组合总和 40.组合总和II 131.分割回文串
|
存储 算法
leetcode-每日一题1252. 奇数值单元格的数目(模拟优化)
时间复杂度:O(q * (m + n) + m * n) 其中q表示 indices 数组的长度,m、n为矩阵的行数和列数,遍历 indices 数组都要更新一次行列,总共需要O(q * (m + n))的时间,最后遍历一次矩阵,总共需要O(m * n)的时间
73 0
leetcode-每日一题1252. 奇数值单元格的数目(模拟优化)
|
算法 C++
<<算法很美>>——(四)——深入递归<二>——“逐步生成结果“类问题之非数值型
<<算法很美>>——(四)——深入递归<二>——“逐步生成结果“类问题之非数值型
<<算法很美>>——(四)——深入递归<二>——“逐步生成结果“类问题之非数值型
代码随想录刷题|Leetcode 39. 组合总和 40.组合总和II 131.分割回文串
代码随想录刷题|Leetcode 39. 组合总和 40.组合总和II 131.分割回文串
代码随想录刷题|Leetcode 39. 组合总和 40.组合总和II 131.分割回文串
【每日一题Day101】LC1664生成平衡数组的方案数 | 预处理+枚举
给你一个整数数组 nums 。你需要选择 恰好 一个下标(下标从 0 开始)并删除对应的元素。请注意剩下元素的下标可能会因为删除操作而发生改变。
79 0