别再用双层遍历循环来做新旧数组对比,寻找新增元素了!使用array.includes和Set来提升代码可读性

简介: 这类问题的重点在于能不能突破基础思路,突破基础思路是从程序员入门变成中级甚至高级的第一步,如果所有需求都通过最基础的业务逻辑来做,是得不到成长的。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~


一、双层循环遍历

1.1、双循环错误示范

       前几天看项目,发现有个新旧数组对比,寻找新增元素的需求竟然是用for写的双循环。大概就像下面这样:

// 假设这是两个数组的真实数据
const array1 = [1, 2, 3, 4, 5, 6, 7, 8];
const array2 = [4, 5, 6, 7, 8];
// 用来存储不重复的元素
const uniqueElements = [];
// 双重循环遍历两个数组
for (let i = 0; i < array1.length; i++) {
    for (let j = 0; j < array2.length; j++) {
        // 检查array1中的元素是否不在array2中
        if (array1[i] !== array2[j]) {
            uniqueElements.push(array1[i]);
        }
    }
}
// 打印不重复的元素
console.log(uniqueElements);

image.gif

       太抽象了,这样代码的?给我一种21世纪用木头刀枪打猎的感觉。还有的是这样写的:

// 假设这是两个数组的真实数据
const array1 = [1, 2, 3, 4, 5, 6, 7, 8];
const array2 = [4, 5, 6, 7, 8];
// 用来存储不重复的元素
let uniqueElements = [];
// 遍历array1中的每个元素
array1.forEach(item1 => {
  // 用另一个forEach来检查array1中的元素是否不在array2中
  let isUnique = array2.forEach((item2, index, array) => {
    return item1 === item2;
  });
  // 如果isUnique为false,说明item1不在array2中
  if (!isUnique) {
    uniqueElements.push(item1);
  }
});
// 打印不重复的元素
console.log(uniqueElements); // 输出: [1, 2, 3]

image.gif

       可是有什么区别吗?forEach看起来是高级一点,但是没有摆脱最基础的逻辑啊,重点是双循环很干,一点都不巧妙。基本的逻辑就是在满足功能、需求和时间效率的基础上,要尽可能少用循环,少用回调,大幅提高代码的可读性和可维护性。

1.2、正确的做法

①使用array.includes()

       最基本的就是要会用array.includes()方法,可以少一次循环。

       在这段代码中,我们使用 forEach 方法遍历 array1 中的每个元素。对于 array1 中的每个元素 item1,我们使用 includes 方法检查它是否不在 array2 中。如果 item1 不在 array2 中,我们就将它添加到 uniqueElements 数组中。最后,我们打印出 uniqueElements 数组,它包含了 array1 中不在 array2 中的所有元素。

// 假设这是两个数组的真实数据
const array1 = [1, 2, 3, 4, 5, 6, 7, 8];
const array2 = [4, 5, 6, 7, 8];
// 用来存储不重复的元素
let uniqueElements = [];
// 遍历array1中的每个元素
array1.forEach(item1 => {
  // 检查array1中的元素是否不在array2中
  if (!array2.includes(item1)) {
    uniqueElements.push(item1);
  }
});
// 打印不重复的元素
console.log(uniqueElements); // 输出: [1, 2, 3]

image.gif

②使用set

       使用集合先去重,然后通过Set.has()方法来判断新增元素。

       在这个代码中,我们首先创建了一个 Set 对象 set2 来存储 array2 中的所有元素。然后,我们使用 forEach 方法遍历 array1 中的每个元素 item1。对于 array1 中的每个元素,我们检查它是否不在 set2 中。如果不在,我们将其添加到 uniqueElements 数组中。最后,我们打印出 uniqueElements 数组,它包含了 array1 中不在 array2 中的所有元素。

// 假设这是两个数组的真实数据
const array1 = [1, 2, 3, 4, 5, 6, 7, 8];
const array2 = [4, 5, 6, 7, 8];
// 创建一个集合来存储array2中的元素,以便快速查找
const set2 = new Set(array2);
// 用来存储不重复的元素
const uniqueElements = [];
array1.forEach(item1 => {
  // 检查item1是否不在array2的集合中
  if (!set2.has(item1)) {
    uniqueElements.push(item1);
  }
});
// 打印不重复的元素
console.log(uniqueElements); // 应该输出: [1, 2, 3]

image.gif

       这样是不是优雅很多?并且时间效率上也有提高。

       既然谈到这里,就简单聊一下array.includes()吧。

二、array.includes()的使用与技巧

image.gif 编辑

2.1、基本语法

arr.includes(searchElement[, fromIndex])

  • searchElement:需要检查是否包含在数组中的元素。
  • fromIndex(可选):开始搜索的索引位置,默认值为 0。如果该值大于数组长度,则返回 false 并且不执行搜索。

2.2、返回值

  • 如果 searchElement 存在于数组中,返回 true。
  • 如果 searchElement 不存在于数组中,返回 false。

2.3、使用技巧

       array.includes()用于判断一个数组是否包含一个指定的值,根据情况返回 truefalse

2.3.1、用户输入验证

       在表单验证中,我们可能需要检查用户输入是否包含某些特定的字符或单词。

const userInput = "Hello, World!";
const bannedWords = ["badword", "offensiveword"];
if (bannedWords.some(word => userInput.includes(word))) {
  console.log("输入包含禁止词汇!");
} else {
  console.log("输入有效。");
}

image.gif

2.3.2、权限检查

       在权限管理系统中,includes() 可以用来检查用户是否具有执行特定操作的权限。

const userPermissions = ["read", "write", "delete"];
const requiredPermission = "delete";
if (userPermissions.includes(requiredPermission)) {
  console.log("用户有权限执行删除操作。");
} else {
  console.log("用户没有权限执行删除操作。");
}

image.gif

2.4、兼容问题

       array.includes() 方法在现代浏览器中得到广泛支持,包括最新版本的 Chrome、Firefox、Safari、Edge 和 Opera。对于不支持 includes() 的旧浏览器,可以使用 Array.prototype.indexOf() 方法作为替代:

if (!Array.prototype.includes) {
  Array.prototype.includes = function(searchElement /*, fromIndex*/)
  {
    'use strict';
    var O = Object(this);
    var len = parseInt(O.length, 10) || 0;
    if (len === 0) {
      return false;
    }
    var fromIndex = !fromIndex || fromIndex < 0 ? 0 : Math.floor(fromIndex);
    for (; fromIndex < len; fromIndex++) {
      if (fromIndex in O && O[fromIndex] === searchElement) {
        return true;
      }
    }
    return false;
  };
}

image.gif

三、总结

       这类问题的重点在于能不能突破基础思路,突破基础思路是从程序员入门变成中级甚至高级的第一步,如果所有需求都通过最基础的业务逻辑来做,是得不到成长的。

       丰富的前端内容请看:各种前端问题的技巧和解决方案

       博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~

相关文章
|
24天前
|
人工智能 Java
Java 中数组Array和列表List的转换
本文介绍了数组与列表之间的相互转换方法,主要包括三部分:1)使用`Collections.addAll()`方法将数组转为列表,适用于引用类型,效率较高;2)通过`new ArrayList&lt;&gt;()`构造器结合`Arrays.asList()`实现类似功能;3)利用JDK8的`Stream`流式计算,支持基本数据类型数组的转换。此外,还详细讲解了列表转数组的方法,如借助`Stream`实现不同类型数组间的转换,并附带代码示例与执行结果,帮助读者深入理解两种数据结构的互转技巧。
Java 中数组Array和列表List的转换
|
1月前
|
JavaScript 前端开发 API
JavaScript中通过array.map()实现数据转换、创建派生数组、异步数据流处理、复杂API请求、DOM操作、搜索和过滤等,array.map()的使用详解(附实际应用代码)
array.map()可以用来数据转换、创建派生数组、应用函数、链式调用、异步数据流处理、复杂API请求梳理、提供DOM操作、用来搜索和过滤等,比for好用太多了,主要是写法简单,并且非常直观,并且能提升代码的可读性,也就提升了Long Term代码的可维护性。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
1月前
|
前端开发 JavaScript 数据格式
通过array.reduce()实现数据汇总、条件筛选和映射、对象属性的扁平化、转换数据格式、聚合统计、处理树结构数据和性能优化,reduce()的使用详解(附实际应用代码)
array.reduce()可以用来数据汇总、条件筛选和映射、对象属性的扁平化、转换数据格式、聚合统计、处理树结构数据、性能优化等,使用难度相对高一些,但是能大大减少代码量。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时
|
1月前
|
移动开发 运维 供应链
通过array.some()实现权限检查、表单验证、库存管理、内容审查和数据处理;js数组元素检查的方法,some()的使用详解,array.some与array.every的区别(附实际应用代码)
array.some()可以用来权限检查、表单验证、库存管理、内容审查和数据处理等数据校验工作,核心在于利用其短路机制,速度更快,节约性能。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
1月前
|
供应链 JavaScript 前端开发
通过array.every()实现数据验证、权限检查和一致性检查;js数组元素检查的方法,every()的使用详解,array.some与array.every的区别(附实际应用代码)
array.every()可以用来数据验证、权限检查、一致性检查等数据校验工作,核心在于利用其短路机制,速度更快,节约性能。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
11月前
|
Python
使用array()函数创建数组
使用array()函数创建数组。
215 3
|
1月前
|
数据采集 JavaScript 前端开发
JavaScript中通过array.filter()实现数组的数据筛选、数据清洗和链式调用,JS中数组过滤器的使用详解(附实际应用代码)
用array.filter()来实现数据筛选、数据清洗和链式调用,相对于for循环更加清晰,语义化强,能显著提升代码的可读性和可维护性。博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
4月前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
134 67
|
6月前
|
人工智能 前端开发 JavaScript
拿下奇怪的前端报错(一):报错信息是一个看不懂的数字数组Buffer(475) [Uint8Array],让AI大模型帮忙解析
本文介绍了前端开发中遇到的奇怪报错问题,特别是当错误信息不明确时的处理方法。作者分享了自己通过还原代码、试错等方式解决问题的经验,并以一个Vue3+TypeScript项目的构建失败为例,详细解析了如何从错误信息中定位问题,最终通过解读错误信息中的ASCII码找到了具体的错误文件。文章强调了基础知识的重要性,并鼓励读者遇到类似问题时不要慌张,耐心分析。
139 5
|
6月前
|
存储 Java
Java“(array) <X> Not Initialized” (数组未初始化)错误解决
在Java中,遇到“(array) &lt;X&gt; Not Initialized”(数组未初始化)错误时,表示数组变量已被声明但尚未初始化。解决方法是在使用数组之前,通过指定数组的大小和类型来初始化数组,例如:`int[] arr = new int[5];` 或 `String[] strArr = new String[10];`。
170 2

热门文章

最新文章

  • 1
    Java 中数组Array和列表List的转换
    46
  • 2
    JavaScript中通过array.map()实现数据转换、创建派生数组、异步数据流处理、复杂API请求、DOM操作、搜索和过滤等,array.map()的使用详解(附实际应用代码)
    78
  • 3
    通过array.reduce()实现数据汇总、条件筛选和映射、对象属性的扁平化、转换数据格式、聚合统计、处理树结构数据和性能优化,reduce()的使用详解(附实际应用代码)
    160
  • 4
    通过array.some()实现权限检查、表单验证、库存管理、内容审查和数据处理;js数组元素检查的方法,some()的使用详解,array.some与array.every的区别(附实际应用代码)
    42
  • 5
    通过array.every()实现数据验证、权限检查和一致性检查;js数组元素检查的方法,every()的使用详解,array.some与array.every的区别(附实际应用代码)
    30
  • 6
    多维数组操作,不要再用遍历循环foreach了!来试试数组展平的小妙招!array.flat()用法与array.flatMap() 用法及二者差异详解
    35
  • 7
    Array.forEach实战详解:简化循环与增强代码可读性;Array.forEach怎么用;面对大量数据时怎么提高Array.forEach的性能
    26
  • 8
    深入理解 JavaScript 中的 Array.find() 方法:原理、性能优势与实用案例详解
    57
  • 9
    JavaScript 中通过Array.sort() 实现多字段排序、排序稳定性、随机排序洗牌算法、优化排序性能,JS中排序算法的使用详解(附实际应用代码)
    125
  • 10
    JavaScript中通过array.filter()实现数组的数据筛选、数据清洗和链式调用,JS中数组过滤器的使用详解(附实际应用代码)
    57