Skip to content

数组

统计信息:字数 6504 阅读14分钟

假设:一维数组,每一项是合法数值,不需要数据类型验证。

数组已经排序,二分法 或者 双指针。

数组没有排序,使用 Hash 记录出现的次数;或者先排序。

1、数组去重

可以使用Hash去重,返回去重后的数组

function unique(arr) {
  let hashTable = {};
  let res = [];
  for (let i = 0; i < arr.length; i++) {
    let item = arr[i]
    if (!hashTable[item]) {
      res.push(item);
      hashTable[item] = true;
    }
  }
  return res;
}

console.log(unique([0,0,1,1,2,2,3]));

可以使用 ES6 的 Set 去重,然后转换成数组返回

function unique(arr) {
  return Array.from(new Set(arr));
  // 简化写法 return [...new Set(arr)];
}

2、数组最值

可以使用循环找最大值

function findMax(arr) {
  if (arr.length === 1) {
    return arr[0];
  }
  let max = arr[0];
  for (let i = 1; i < arr.length; i++) {
    if (max < arr[i]) {
      max = arr[i];
    }
  }
  return max;
}

可以使用Math对象API求最大值

function findMax(arr) {
  return Math.max(...arr);
}

3、其他几种数组去重(了解)

// 方法一:
// 双层循环,外层循环元素,内层循环时比较值
// 如果有相同的值则跳过,不相同则push进数组
Array.prototype.distinct = function() {
  var arr = this, result = [], i, j;
  var len = arr.length;
  for (i = 0; i < len; i++) {
    for (j = i + 1; j < len; j++) {
      if (arr[i] === arr[j]) {
        j = ++i;
      }
    }
    result.push(arr[i]);
  }
  return result;
}
var arra = [1, 2, 3, 4, 4, 1, 1, 2, 1, 1, 1];
arra.distinct(); //返回[3,4,2,1]

// 方法二:利用splice直接在原数组进行操作
// 双层循环,外层循环元素,内层循环时比较值,值相同时,则删去这个值
// 注意点: 删除元素之后,需要将数组的长度也减1.
Array.prototype.distinct = function() {
  var arr = this i, j;
  var len = arr.length;
  for (i = 0; i < len; i++) {
    for (j = i + 1; j < len; j++) {
      if (arr[i] == arr[j]) {
        arr.splice(j, 1);
        len--;
        j--;
      }
    }
  }
  return arr;
};
var a = [1, 2, 3, 4, 5, 6, 5, 3, 2, 4, 56, 4, 1, 2, 1, 1, 1, 1, 1, 1, ];
var b = a.distinct();
console.log(b.toString()); //1,2,3,4,5,6,56
// 优点:简单易懂
// 缺点:占用内存高,速度慢

// 方法三:利用对象的属性不能相同的特点进行去重
Array.prototype.distinct = function() {
  var arr = this, result = [], i, j;
  var len = arr.length;
  for (i = 0; i < arr.length; i++) {
    if (!obj[arr[i]]) { //如果能查找到,证明数组元素重复了
      obj[arr[i]] = 1;
      result.push(arr[i]);
    }
  }
  return result;
};
var a = [1, 2, 3, 4, 5, 6, 5, 3, 2, 4, 56, 4, 1, 2, 1, 1, 1, 1, 1, 1, ];
var b = a.distinct();
console.log(b.toString()); //1,2,3,4,5,6,56

// 方法四:数组递归去重
// 运用递归的思想,先排序,然后从最后开始比较,遇到相同,则删除
Array.prototype.distinct = function() {
  var arr = this;
    var len = arr.length;
  arr.sort(function(a, b) { //对数组进行排序才能方便比较
    return a - b;
  });
  function loop(index) {
    if (index >= 1) {
      if (arr[index] === arr[index - 1]) {
        arr.splice(index, 1);
      }
      loop(index - 1); //递归loop函数进行去重
    }
  }
  loop(len - 1);
  return arr;
};
var a = [1, 2, 3, 4, 5, 6, 5, 3, 2, 4, 56, 4, 1, 2, 1, 1, 1, 1, 1, 1, 56, 45, 56];
var b = a.distinct();
console.log(b.toString()); //1,2,3,4,5,6,45,56

// 方法五:利用indexOf以及forEach
Array.prototype.distinct = function() {
  var arr = this,
    result = [],
    len = arr.length;
  arr.forEach(function(v, i, arr) { //这里利用map,filter方法也可以实现
    var bool = arr.indexOf(v, i + 1); //从传入参数的下一个索引值开始寻找是否存在重复
    if (bool === -1) {
      result.push(v);
    }
  })
  return result;
};
var a = [1, 1, 3, 3, 3, 2, 3, 3, 2, 2, 1, 23, 1, 23, 2, 3, 2, 3, 2, 3];
var b = a.distinct();
// console.log(b.toString()); //1,23,2,3

4、合并多个数组,并去重

// 一、 concat() 方法
// 思路:concat() 方法将传入的数组或非数组值与原数组合并, 组成一个新的数组并返回。该方法会产生一个新的数组。
function concatArr(arr1, arr2) {
  var arr = arr1.concat(arr2);
  arr = unique1(arr); //再引用上面的任意一个去重方法
  return arr;
}

// 二、 Array.prototype.push.apply()
// 思路:该方法优点是不会产生一个新的数组。
var a = [1, 2, 3];
var b = [4, 5, 6];
Array.prototype.push.apply(a, b); //a=[1,2,3,4,5,6]
// 等效于:a.push.apply(a, b);
// 也等效于[].push.apply(a, b); 
function concatArray(arr1, arr2) {
  Array.prototype.push.apply(arr1, arr2);
  arr1 = unique1(arr1);
  return arr1;
}

5、获取数组中重复元素

思路1,遍历一次数组,判断当前元素出现过的次数,如果出现多次,放到结束中(indexOf 性能较差)

function getRepeatItem (arr) {
  let result = [];
  for (let i =0; i < arr.length; i++) {
    if (arr.indexOf(arr[i]) !== arr.lastIndexOf(arr[i]) && result.indexOf(arr[i]) === -1) {
      result.push(arr[i]);
    }
  }
  return result;
}

// getRepeatItem([1,2,3,4,3,2]);

思路2,遍历一次数组,使用 Hash 记录出现过的元素,如果出现次数是2,那么放入结果数组。比思路1性能好。

function getRepeatItem (arr) {
  let result = [];
  let dict = {};
  for (let i = 0; i < arr.length; i++) {
    if (dict[arr[i]]) {
      dict[arr[i]]++;
    } else {
      dict[arr[i]] = 1;
    }
    if (dict[arr[i]] === 2) {
      result.push(arr[i]);
    }
  }
  return result;
}

6、数组集合运算(交集,并集,补集,差集)

如果每一个数组具有重复值,首先数组去重;

直接将两个数组合并成一个数组,然后求合并后数组中的重复元素,就是数组的交集。

或者遍历其中一个数组,然后在第二个数组中indexOf,如果不是-1那么就返回新的数组。

function getIntersection(arr1, arr2) {
  let arr3 = arr1.join(arr2);
  return getRepeatItem(arr3);
}

function getIntersection(arr1, arr2) {
  let result = [];
    for (let i = 0; i < arr1.length; i++) {
    if (arr2.indexOf(arr1[i]) && result.indexOf(arr1[i] === -1)) {
      result.push(arr1[i]);
    }
  }
  return result;
}

getIntersection([1,2,3,4], [1,2,5,6]);

Last update: November 9, 2024