数组¶
统计信息:字数 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