比较JS合并数组的各种方法及其优劣

本文属于JavaScript的基础技能. 我们将学习结合/合并两个JS数组的各种常用方法,并比较各种方法的优缺点.

我们先来看看具体的场景:


  1. var q = [ 5, 5, 1, 9, 9, 6, 4, 5, 8];
  2. var b = [ "tie", "mao", "csdn", "ren", "fu", "fei" ];


很明显,数组 q 和 b 简单拼接的结果是:

  1. [
  2.     5, 5, 1, 9, 9, 6, 4, 5, 8, 
  3.     "tie", "mao", "csdn", "ren", "fu", "fei"
  4. ]


concat(..)方法

最常见的用法如下:


  1. var c = q.concat( b );
  2. q; 
  3. // [5,5,1,9,9,6,4,5,8]
  4. b; 
  5. // ["tie","mao","csdn","ren","fu","fei"];
  6. c; 
  7. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]


如您所见, c 是一个全新的数组, 表示 q 和 b 这两个数组的组合, 但是 q 和 b 现在没用了是吧?

如果 q 数组有10000个元素, b 数组也有有10000个元素? 那么数组c现在就有20000个元素, 这种方式占用了2倍的内存.
“这没问题!”,你可能会觉得. 只要将 q 和 b 置空就行, 然后就会被垃圾回收,对吗?问题解决了!

  1. q = b = null; 
  2. // `q` and `b` 现在可以被垃圾回收了


额? 如果数组都很小,那自然没问题. 但对大型的数组,或需要多次重复处理时, 内存就被限制了, 它还需要进行优化.

循环插入

OK, 让我们把一个数组的内容加入到另一个中试试,使用 Array#push() 方法:

  1. // 将数组 `b` 插入 `q`
  2. for (var i=0; i < b.length; i++) {
  3.     q.push( b[i] );
  4. }
  5. q; 
  6. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]
  7. b = null;


现在, q中存放了两个原始数组的内容(q + b).

看样子对内存优化做的不错.

但如果 q 数组很小而 b 又很大呢? 出于内存和速度的考虑,这时想把较小的 q 插入到 b 前面. 没问题,只要用unshift() 方法代替 push() 即可, 对应的也要从大到小进行循环遍历:


  1. // `q` into `b`:
  2. for (var i=q.length-1; i >= 0; i--) {
  3.     b.unshift( q[i] );
  4. }
  5. b; 
  6. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]
  7. q = null;


实用技巧

悲催的是,for循环很土并且难以维护. 我们能做得更好吗?

我们先试试 Array#reduce :


  1. // `b` onto `q`:
  2. q = b.reduce( function(coll,item){
  3.     coll.push( item );
  4.     return coll;
  5. }, q );
  6. q; 
  7. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]
  8. // or `q` into `b`:
  9. b = q.reduceRight( function(coll,item){
  10.     coll.unshift( item );
  11.     return coll;
  12. }, b );
  13. b; 
  14. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]


Array#reduce() 和 Array#reduceRight() 很高大上,但有点笨重,而且一般人也记不住.  JS规范6 中的 => 箭头函数(arrow-functions) 能让代码量大大减少, 但需要对每个数组元素执行函数调用, 也是很渣的手段.

那么下面的代码怎么样呢?


  1. // `b` onto `q`:
  2. q.push.apply( q, b );
  3. q; 
  4. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]
  5. // or `q` into `b`:
  6. b.unshift.apply( b, q );
  7. b; 
  8. // [5,5,1,9,9,6,4,5,8,"tie","mao","csdn","ren","fu","fei"]


BIG更高了,是吧!? 特别是 unshift() 方法不需要像前面那样考虑相反的顺序. ES6 的展开运算符(spread operator, 加 … 前缀)就更高端了: a.push( …b ) 或者 b.unshift( …a )

但是,事实上这种方法还是太乐观了. 在这两种情况下,不管是将 a 或 b 传递给 apply() 作为第二个参数(apply方式调用Function时第一个参数在内部变成this,即context,上下文,作用域), 还是使用 … 展开运算符的方式, 实际上数组都会被打散成为函数的 arguments .

第一个主要的问题是,占用了双倍的内存(当然,是临时的!),因为需要将数组复制到函数栈之中. 此外,不同的JS引擎有不同的实现算法,可能会限制了函数可以传递的参数数量.

如果数组添加了一百万个元素, 那一定会超过函数栈所允许的大小, 不管是push() 或 unshift()调用. 这种方式只在几千个元素时可用,所以必须限制其不能超过一定范围.

注意: 你也可以试试 splice(), 肯定会发现他和 push(..)/unshift(..) 都是一样的限制.

一种选择是继续使用这种方法,但是采用分批次处理:


  1. function combineInto(q,b) {
  2.     var len = q.length;
  3.     for (var i=0; i < len; i=i+5000) {
  4.         
  5. // 一次处理5000条
  6.         b.unshift.apply( b, q.slice( i, i+5000 ) );
  7.     }
  8. }


等等,我们损害了代码的可读性(甚至是性能!). 在我们放弃之前结束这个旅程吧.

总结

Array#concat() 是久经考验的方法, 用于组合两个(或多个)数组. 但他创建了一个新的数组,而不是修改现有的一个.有很多变通的手法,但他们都有不同的优缺点,需要根据实际情况来选择.

上面列出了各种 优点/缺点,也许最好的(包括没有列出的)方法是 reduce(..) 和 reduceRight(..) 无论你选择什么,都应该批判性地思考你的数组合并策略,而不是把它当作理所当然的事情.

本文来源:www.xzzxxq.com

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 系统安全测试的常用方法包括静态测试、动态测试、白盒测试、黑盒测试等。静态测试主要是在安全分析阶段,通过分析系统设计文档和程序代码来确定系统是否具备安全特性;动态测试主要是在安全检测阶段,通过执行和观察系统的行为来确定系统是否具备安全特性;白盒测试主要是在安全检测阶段,通过分析系统的内部构造和源码来确定系统是否具备安全特性;黑盒测试主要是在安全检测阶段,通过系统输入和输出来确定系统是否具备安全特性。不同的测试方法各有优劣,根据实际情况,可以综合运用,从而获得更好的测试效果。 ### 回答2: 系统安全测试是为了评估计算机系统或软件在面临各种威胁和攻击时的安全性能和强度。常用的系统安全测试方法主要有黑盒测试、白盒测试和灰盒测试。 黑盒测试是在不考虑系统内部实现细节的情况下进行的测试。测试人员将系统视为一个黑盒,只关注输入输出以及系统对输入的处理,不关心系统实现细节。黑盒测试的优点是测试人员不需要了解系统的内部结构和代码,只需关注功能是否按照预期运行。缺点是不能完全覆盖系统内部的潜在安全漏洞,可能无法发现代码中的错误和不安全设计。 白盒测试是基于对系统内部结构和代码的理解进行的测试。测试人员可以通过分析系统的内部实现来发现潜在的安全漏洞和风险。白盒测试的优点是能够深入了解系统的内部结构,找出隐藏的安全问题,提高测试的准确性。缺点是需要测试人员具备丰富的技术知识和编程经验,且对系统的了解程度要求较高。 灰盒测试是黑盒和白盒测试的结合。测试人员了解系统的部分内部实现细节,但不完全了解所有细节。灰盒测试的优点是兼顾了黑盒测试的功能覆盖和白盒测试的深入分析,能够发现系统中的潜在问题和安全漏洞。缺点是对测试人员的技术能力和知识要求较高,且测试的细节和深度因人而异。 综上所述,黑盒测试注重系统功能的测试,白盒测试注重系统内部细节的分析,灰盒测试结合了两者的优点。不同的测试方法具有不同的优劣势,选择适合的测试方法取决于测试的目的、时间和资源的限制,以及测试人员的技术水平和经验。 ### 回答3: 系统安全测试是为了评估计算机系统的安全性能和漏洞而进行的一系列测试。常用的系统安全测试方法包括渗透测试、漏洞扫描、安全配置审计等。 首先,渗透测试是一种模拟黑客攻击的方法,通过模拟实际攻击行为,尝试找到系统的弱点和漏洞。渗透测试的优势是能够全面检测系统的安全性,找出可能存在的风险和漏洞,并提供有效的修复建议。然而,渗透测试需要有专业的技术人员进行,并且可能会对系统造成一定程度的影响,如造成系统崩溃或数据泄漏的风险。 其次,漏洞扫描是一种自动化的工具,用于检测系统中已知的安全漏洞。漏洞扫描的优势在于其高效性和快速性,能够快速扫描整个系统,并给出可能存在的漏洞报告。然而,漏洞扫描可能会漏报或误报漏洞,因为只能针对已知的漏洞进行扫描,对未知的安全漏洞无法有效发现。 最后,安全配置审计方法通过检查系统的安全配置是否符合最佳实践来评估系统的安全性。它可以发现由于安全配置不当而导致的潜在风险。安全配置审计的优势在于其简单易行,可以快速发现系统的安全配置问题。然而,它只能检查已知的安全配置问题,无法检测未知的安全漏洞,也无法评估系统的实际安全性。 综上所述,渗透测试能够全面评估系统的安全性,但需要专业技术人员进行,并可能对系统造成一定风险;漏洞扫描能够快速检测已知漏洞,但存在漏报和误报的可能性;安全配置审计简单易行,但无法检测未知漏洞。因此,在系统安全测试中,可以根据实际情况综合使用这些方法,以达到更全面和准确的评估结果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值