自定义工具函数库(二) 数组相关
最终仓库:utils: 自定义工具库
以前的笔记:JS数组常用的方法
1. 数组声明式系列方法
1.1 map函数封装实现
map() 方法创建一个新数组,其结果是该数组中的每个元素各自调用一次提供的函数后的返回值
循环,数组的每个元素都调用一次函数,并把每次循环得到的返回值都存好,循环结束后,把存好的数组返回。
function map(arr, callback) {
let ret = []
for (let i = 0; i < arr.length; i++) {
ret.push(callback(arr[i], i))
}
return ret
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>map函数封装实现</title>
<script src="./map.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4, 5]
const result = map(arr, (item, index) => {
console.log(index)
return item + 1
})
console.log(result)
</script>
</body>
</html>
1.2 reduce函数
对数组中的每个元素执行一个由您提供的reducer函数(升序执行),将其结果汇总为单个返回值。
循环遍历数组,并每次都把调用函数得到的值,重新赋值给ret 变量,然后作为下一次调用函数时的第一个参数
function reduce(arr, callback, initValue) {
let ret = initValue
for (let item of arr) {
ret = callback(ret, item)
}
return ret
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>reduce函数封装实现</title>
<script src="./reduce.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4]
const result = reduce(arr,
(preValue, curValue) => preValue + curValue,
10)
console.log(result)
</script>
</body>
</html>
1.3 filter函数
创建一个新数组, 其包含通过所提供函数实现的测试的所有元素。
循环,如果以该数组元素作为参数调用函数的返回值为true ,则存好。
function filter(arr, callback) {
let ret = []
for (let i = 0; i < arr.length; i++) {
if (callback(arr[i], i)) {
ret.push(arr[i])
}
}
return ret
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>filter函数封装实现</title>
<script src="./filter.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4, 5]
const result = filter(arr, (item, index) => index % 2 === 0)
console.log(result)
</script>
</body>
</html>
1.4 find函数
返回数组中满足提供的测试函数的第一个元素的值。否则返回 undefined 。
function find(arr, callback) {
for (let i = 0; i < arr.length; i++) {
if (callback(arr[i], i)) {
return arr[i]
}
}
return undefined
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>find函数封装实现</title>
<script src="./find.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4, 5]
const result1 = find(arr, (item, index) => item >= 3)
console.log(result1)
const result2 = find(arr, (item, index) => item >= 10)
console.log(result2)
</script>
</body>
</html>
1.5 findIndex函数
返回数组中满足提供的测试函数的第一个元素的索引。若没有找到对应元素则返回-1。
和find()函数类似,只不过是返回满足条件的第一个元素的索引,以及没有满足的是返回-1
function findIndex(arr, callback) {
for (let i = 0; i < arr.length; i++) {
if (callback(arr[i], i)) {
return i
}
}
return -1
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>findIndex函数封装实现</title>
<script src="./findIndex.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4, 5]
const result1 = findIndex(arr, (item, index) => item >= 3)
console.log(result1)
const result2 = findIndex(arr, (item, index) => item >= 10)
console.log(result2)
</script>
</body>
</html>
1.6 every函数
测试一个数组内的所有元素是否都能通过某个指定函数的测试。它返回一个布尔值。
function every(arr, callback) {
for (let i = 0; i < arr.length; i++) {
if (!callback(arr[i], i)) {
return false
}
}
return true
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>every函数封装实现</title>
<script src="./every.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4, 5]
const result1 = every(arr, (item, index) => item >= 3)
console.log(result1)
const result2 = every(arr, (item, index) => item < 10)
console.log(result2)
</script>
</body>
</html>
1.7 some函数
测试数组中是不是至少有1个元素通过了被提供的函数测试。它返回的是一个Boolean类型的值。
function some(arr, callback) {
for (let i = 0; i < arr.length; i++) {
if (callback(arr[i], i)) {
return true
}
}
return false
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>some函数封装实现</title>
<script src="./some.js"></script>
</head>
<body>
<script>
const arr = [1, 2, 3, 4, 5]
const result1 = some(arr, (item, index) => item < 3)
console.log(result1)
const result2 = some(arr, (item, index) => item < 0)
console.log(result2)
</script>
</body>
</html>
2. 数组去重
2.1 forEach() + indexOf()
function unique(arr) {
const ret = []
arr.forEach(item => {
if (ret.indexOf(item) === -1) {
ret.push(item)
}
})
return ret
}
2.2 forEach() + 容器(对象)
function unique(arr) {
const ret = []
const obj = {}
arr.forEach(item => {
if (obj[item]) {
return
}
obj[item] = true
ret.push(item)
})
return ret
}
2.3 Set + Array.from() 或 扩展运算符(…)
function unique(arr) {
let set = new Set(arr)
return [...set]
}
3. concat函数
用于合并两个或多个数组。此方法不会更改现有数组,而是返回一个新数组。
function concat(arr, ...args) {
const ret = [...arr]
args.forEach(item => {
if (Array.isArray(item)) {
ret.push(...item)
} else {
ret.push(item)
}
})
return ret
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>concat函数实现</title>
<script src="./concat.js"></script>
</head>
<body>
<script>
let arr = [1, 2, 3]
const result = concat(arr, [4, 5, 6], 7, 8, [9, 10])
console.log(result)
</script>
</body>
</html>
4. slice数组切片
返回一个新的数组对象,这一对象是一个由 begin 和 end 决定的原数组的浅拷贝(包括 begin ,不包括end )。原始数组不会被改变。
function slice(arr, begin, end) {
if (arr.length === 0) {
return []
}
begin = begin || 0
if (begin >= arr.length) {
return []
}
end = end || arr.length
const ret = []
for (let i = 0; i < arr.length; i++) {
if (i >= begin && i < end) {
ret.push(arr[i])
}
}
return ret
}
测试:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>slice函数实现</title>
<script src="./slice.js"></script>
</head>
<body>
<script>
let arr = [1, 2, 3, 4, 5, 6, 7]
const result1 = slice(arr, 2, 6)
console.log(result1)
const result2 = slice(arr)
console.log(result2)
const result3 = slice(arr, 2)
console.log(result3)
const result4 = slice(arr, 3, 1)
console.log(result4)
</script>
</body>
</html>
5. 数组扁平化
数组扁平化是指将一个多维数组变为一维数组
内置方法flat:
let arr = [1, 2, [3, 4],
[
[5], 6
], 7
]
const result = arr.flat(Infinity)
console.log(result)
5.1 递归
function flatten(arr) {
let ret = []
arr.forEach(item => {
if (Array.isArray(item)) {
ret = ret.concat(flatten(item))
} else {
ret = ret.concat(item)
}
})
return ret
}
5.2 扩展运算符…
function flatten(arr) {
let ret = [...arr]
while (ret.some(item => Array.isArray(item))) {
ret = [].concat(...ret)
}
return ret
}
5.3 toString / join + split + map
先通过toString 或join 把数组转成字符串,在通过split 转换成数组,最后还需要通过map 函数,把数组的每一个元素变回数字。(不考虑其他类型的数组的话)
function flatten(arr) {
return arr.toString().split(',').map(item => Number(item))
}
5.4 reduce
遍历数组,如果是数组,则递归遍历,否则,通过concat 拼接起来
function flatten(arr) {
return arr.reduce((pre, cur) => {
return pre.concat(Array.isArray(cur) ? flatten(cur) : cur)
}, [])
}
6. 数组分块
语法:chunk(array, size)
功能:将数组拆分成多个size大小长度的区块,每个区块组成小数组,整体组成一个二维数组
例子:[1, 2, 3, 4, 5, 6]调用chunk(arr, 4) => [[1, 2, 3, 4], [5, 6]]
function chunk(arr, size = 1) {
let ret = []
let temp = []
arr.forEach(item => {
if (temp.length === 0) {
ret.push(temp)
}
temp.push(item)
if (temp.length === size) {
temp = []
}
})
return ret
}
7. 数组差集
- 语法: difference(arr1, arr2)
- 功能: 得到当前数组中所有不在arr中的元素组成的数组(不改变原数组)
- 例子: difference([1,3,5,7], [5, 8]) ==> [1, 3, 7]
function difference(arr1, arr2 = []) {
if (arr1.length === 0) {
return []
}
if (arr2.length === 0) {
return arr1.slice()
}
let ret = arr1.filter(item => !arr2.includes(item))
return ret
}
|