由JS数组去重说起
一、问题描述:
var array=[1,45,3,1,4,67,45],请编写一个函数reDup来去掉其中的重复项,即
reDup(array);
console.log(array);//[1,45,3,4,67]
二、遍历的思路
首先,我们很自然想到,最终array数组得到了改变,我们可以使用引用类型的按值传递在内部改变arr的值。
因此,我们可以加入一个形式参数:
function reDup(arr){
}
当我们执行reDup(array)
时,实际上array
和reDup函数中的形式参数arr
都指向内存中保存的同一个数组。
现在如何去掉其中的重复项呢?
首先,最容易想到的是遍历
:
从arr[0]开始,依次看后面的数组成员是否则与arr[0]相同,若相同,就删除。
第一轮:arr[1]、arr[2]…依次与arr[0]比较;
第二轮:arr[2]、arr[3]…依次与arr[1]比较;
…
第arr.length-1轮:arr[arr.length-2]与arr[arr.length-1]比较。
因此,我们不难写出如下代码:
function reDup(arr){
for(var i=0,len=arr.length;i<len-1;i++)
{
for(var j=i+1;j<len;j++)
{
if(arr[j]===arr[i])
{
arr.splice(j,1);
}
}
}
}
我们测试一下结果是否正确:
var array=[1,45,3,1,4,67,45];
reDup(array);
console.log(array);//[1, 45, 3, 4, 67]
我们看到结果符合预期。
三、遍历的问题
如果我们观察得更仔细一些,就会发现:
当我们进行第一轮和array[0]的比较,到array[3]和array[0]时,发现两者相同,便会删除array[3];
这样我们的数组变成[1,45,3,4,67,45],即原先的array[4]变成 了现在的array[3]。
下一次比较是array[4]和array[0]的比较,即原先的array[5]和array[0]的比较。
我们看到原先的array[4]和array[0]并没有进行比较。即:
每删除一项,就会导致被删除项的后一项未参与比较。
为什么我们的结果是正确的呢?
在我们上面的例子中,被删除项的后一项都与被删除项不相同,因此并不会影响结果,现在我们来看另外的例子:
var array=[1,45,3,1,1,1,4,67,45,45,45];
reDup(array);
console.log(array);//[1, 45, 3, 1, 4, 67, 45]
这里显然不是我们想要的结果。
执行array[0](值为1)与array[3]比较后,将删除array[3],原array[3]后面的项整体会向前移一位,下次比较是array[0]与array[4]的比较,实质上是array[0]和原array[5]的比较。当然,这里会删除原array[5]。但是原array[4]实际上并没有和array[0]比较,由于原array[4]和array[0]是相同的,故导致结果不正确。
我们可以得出结论:
每删除一项,就会导致被删除项的后一项未能参与比较,而更后面一项却又能正常比较。
四、遍历的改进
由上,改进的一种思路是强行让被删除项的后一项参与比较,若被删除项的后一项也与被删除项一致,将再次删除它。
这也导致新的问题:后面的所有项都要向前再移一个位置。
事实上,鉴于删除项的不确定性和数组项的索引位置被改变,我们可以改进为从后往前遍历:
function reDup(arr){
for(var i=0,len=arr.length;i<len-1;i++)
{
for(var j=len-1;j>i;j--)
{
if(arr[j]===arr[i])
{
arr.splice(j,1);
}
}
}
}
var array=[1,45,3,1,1,1,4,67,45,45,45];
reDup(array);
console.log(array);//[1, 45, 3, 4, 67]
结果符合预期。
五、边界条件判定
现在,我们还不能认为万事大吉了。一个问题时,对于一些特殊情况,函数是否也能返回正确值?
首先,假使数组为空数组:
len的值为0,循环一开始就不会执行,因此实际上函数什么也没做,最终还是空数组,结果符合预期。
假使数组仅有1项:
len的值为1,同样循环一开始就不会执行,函数什么也没做,最终还是原数组,结果符合预期。
即使数组有2项:
len的值为2,外层循环执行1次,内层循环也执行1次,结果符合预期。如:
var b=[1,1]; reDup(b); console.log(b);//[1]
如果有更多的项,那么将符合预期,因此能够正确处理边界条件。
六、栈和队列方法:比较巧妙
事实上,我们还可以有另一种思路:
从前往后依次将数组项取出,若该项尚未取出过,则放入数组尾部,然后去掉该项。若已取出过,则直接抛弃。当完成全部取出操作,原数组已变成去重后的数组。
好处是不用使用循环嵌套,实现的代码如下:
function reDup(arr){
var temp=[];
var len=arr.length;
for (var i=0; i < len; i++) {
if (temp.indexOf(arr[0])===-1) {
temp.push(arr[0]);
arr.push(arr[0]);
}
arr.shift();
}
}
var array=[1,45,3,1,1,1,4,67,45,45,45];
reDup(array);
console.log(array);//[1, 45, 3, 4, 67]
同样可以正确处理边界条件。
由JS数组去重说起的更多相关文章
- js数组去重的4种方法
js数组去重,老生长谈,今天对其进行一番归纳,总结出来4种方法 贴入代码前 ,先对浏览器Array对象进行支持indexOf和forEach的polyfill Array.prototype.inde ...
- JS 数组去重(数组元素是对象的情况)
js数组去重有经典的 几种方法 但当数组元素是对象时,就不能简单地比较了,需要以某种方式遍历各值再判断是否已出现. 因为: 1.如果是哈希判断法,对象作哈希表的下标,就会自动转换成字符型类型,从而导致 ...
- js数组去重常用方法
js数组去重是面试中经常会碰到的问题,无论是前端还是node.js数组常见的有两种形式,一种是数组各元素均为基本数据类型,常见的为数组字符串格式,形如['a','b','c'];一种是数组各元素不定, ...
- js 数组去重小技巧
js 数组去重小技巧 Intro 今天遇到一个问题,需要对数据进行去重,想看一下有没有什么比较方便的方法,果然有些收获. Question 问题描述: 我有一个这样的数据: [ { "Pro ...
- JS数组去重的几种常见方法
JS数组去重的几种常见方法 一.简单的去重方法 // 最简单数组去重法 /* * 新建一新数组,遍历传入数组,值不在新数组就push进该新数组中 * IE8以下不支持数组的indexOf方法 * */ ...
- js数组去重五种方法
今天来聊一聊JS数组去重的一些方法,包括一些网上看到的和自己总结的,总共5种方法(ES5). 第一种:遍历数组法 这种方法最简单最直观,也最容易理解,代码如下: var arr = [2, 8, 5, ...
- js数组去重的方法(转)
JS数组去重的几种常见方法 一.简单的去重方法 // 最简单数组去重法 /* * 新建一新数组,遍历传入数组,值不在新数组就push进该新数组中 * IE8以下不支持数组的indexOf方法 * */ ...
- js 数组去重方法汇总
<!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8&quo ...
- js数组去重 javascript版
//js数组去重 //思路: // 1.放入第一个元素 // 2.放入第n个元素,和第n个之前的元素就行比较,如果有重复,则跳过.没有重复就加入数组中 // 3.返回新的去重后数组 Array.pro ...
- js数组去重(多种方法)
// js数组去重 Array.prototype.fun1 = function(){ var arr = this, result = [], i, len = arr.length; for(i ...
随机推荐
- 微信小程序将外部数据从父组件中传入到子组件
小程序组件开发遇到一个组件内嵌两个组件,而这两个子组件所使用的数据来自于同一个API,如下图所示. 如果这时候两个子组件各自导入同一个接口就会显得多余.另外的办法是由父组件导入接口数据,再从父组件将接 ...
- Shell:Day03笔记
编程原理:1.编程结束 驱动 硬件默认是不能使用的 CPU控制硬件 不同的厂家硬件设备之间需要进行指令沟通,就需要驱动程序来进行“翻译” 编程语言的分类: 高级语言.超高级语言需要翻 ...
- SpringMVC(一):简介和第一个程序
本文是按照狂神说的教学视频学习的笔记,强力推荐,教学深入浅出一遍就懂!b站搜索狂神说或点击下面链接 https://space.bilibili.com/95256449?spm_id_from=33 ...
- Array(数组)对象-->概念和创建
1.什么是数组? 数组对象是使用单独的变量名来存储一系列的值. 2.数组创建的三种方法: 方法1:常规方式 var arr=new Array(); arr[0]="lisa"; ...
- STC15F2K60S2串口通信的应用。
前言:由于不可抗拒因素,初始的STC12C5A60S2芯片由于无法进行烧录(...因为没带有锁紧座的开发板),暂且使用STC15F2K60S2芯片.. 一 串行通信概述: 串口通信有SPI IIC U ...
- 3分钟掌握Quartz.net分布式定时任务的姿势
引言 长话短说,今天聊一聊分布式定时任务,我的流水账笔记: ASP.NET Core+Quartz.Net实现web定时任务 AspNetCore结合Redis实践消息队列 细心朋友稍一分析,就知道还 ...
- 事务的传播属性及隔离级别 Spring
事务的传播属性(Propagation) REQUIRED ,这个是默认的属性 Support a current transaction, create a new one if none exis ...
- 【Java】从Null开始,在Windows上下载和安装JDK
下载部分: 方式一: 从官方网站上下载:https://www.oracle.com/java/technologies/javase-downloads.html Oracle已经更新了软件政策,要 ...
- Docker搭建Nessus pro笔记
0x01 准备Docker环境 拉取镜像: docker pull ubuntu 创建容器: docker run -p 9922:22 -p 8834:8834 --name nessus -it ...
- python超实用的30 个简短的代码片段(三)
Python是目前最流行的语言之一,它在数据科学.机器学习.web开发.脚本编写.自动化方面被许多人广泛使用. 它的简单和易用性造就了它如此流行的原因. 如果你正在阅读本文,那么你或多或少已经使用过P ...