JavaScript的Array.prototype.includes方法是什么?怎么用?

javascript 中 array.prototype.includes 方法用于快速判断数组是否包含特定元素,返回布尔值。1. 语法为 arr.includes(searchelement[, fromindex]),searchelement 是查找的元素,fromindex 是可选的起始索引,默认从 0 开始;2. includes 方法直接返回 true 或 false,语义清晰且处理 nan 更可靠,indexof 则因返回索引需额外判断;3. 对于对象或数组等复杂数据类型,includes 使用严格相等(===)比较引用而非内容,需借助 some 方法结合自定义逻辑实现深度比较;4. 在大型数组中频繁使用 includes 性能较差,时间复杂度为 o(n),可用 set 替代,其 has 方法平均时间复杂度为 o(1),适用于查找操作多、数据较稳定的场景。

JavaScript的Array.prototype.includes方法是什么?怎么用?

Array.prototype.includes 方法是 JavaScript 中一个非常实用的工具,它能帮助我们快速判断一个数组是否包含某个特定的元素,并直接返回一个布尔值(truefalse)。这比我们以前用 indexOf 然后判断是否等于 -1 要直观得多。

JavaScript的Array.prototype.includes方法是什么?怎么用?

解决方案

使用 includes 方法非常直接。它的基本语法是 arr.includes(searchElement[, fromIndex])

searchElement:你想要在数组中查找的元素。fromIndex (可选):从数组的哪个索引位置开始搜索。如果省略,默认从索引 0 开始。如果 fromIndex 大于或等于数组的长度,includes 会直接返回 false,不会进行搜索。如果 fromIndex 是负数,它会从 array.length + fromIndex 的位置开始搜索,但仍会搜索整个数组。

举几个例子可能更清晰:

立即学习“Java免费学习笔记(深入)”;

JavaScript的Array.prototype.includes方法是什么?怎么用?

const numbers = [1, 2, 3, 4, 5];console.log(numbers.includes(3)); // trueconsole.log(numbers.includes(6)); // falseconst fruits = ['apple', 'banana', 'cherry', 'apple'];console.log(fruits.includes('banana')); // trueconsole.log(fruits.includes('grape')); // false// 从特定索引开始搜索console.log(fruits.includes('apple', 1)); // true (从索引1开始,找到了第二个'apple')console.log(fruits.includes('apple', 2)); // true (从索引2开始,找到了第二个'apple')console.log(fruits.includes('apple', 4)); // false (索引4超出数组范围,或者说没有元素了)const mixed = [1, 'hello', null, undefined, NaN, 0];console.log(mixed.includes(NaN)); // true (这一点很重要,后面会详细说)console.log(mixed.includes(0)); // trueconsole.log(mixed.includes(null)); // true

includes() 和 indexOf() 有什么区别为什么选择 includes()?

这真是个老生常谈的问题了,但每次讨论都很有意思。从功能上看,includes()indexOf() 都能检查元素是否存在,但它们的侧重点和行为差异,尤其是处理 NaN 的方式,让我更倾向于在判断存在性时使用 includes()

indexOf() 方法返回的是目标元素在数组中找到的第一个索引位置,如果没找到则返回 -1。所以,你通常会看到这样的代码:if (arr.indexOf(element) !== -1) 来判断元素是否存在。

JavaScript的Array.prototype.includes方法是什么?怎么用?

includes() 则直接返回一个布尔值,这让代码意图更加明确:你就是想知道“有没有”,而不是“在哪里”。从语义上讲,arr.includes(element) 读起来更自然,也更符合直觉。

但真正的“杀手锏”区别在于它们对 NaN(Not a Number)的处理。在 JavaScript 中,NaN 是个很特殊的值,它不等于任何值,甚至不等于它自己 (NaN === NaNfalse)。这导致了 indexOf() 在查找 NaN 时会“失灵”:

const weirdNumbers = [1, 2, NaN, 3];console.log(weirdNumbers.indexOf(NaN)); // -1 (indexOf 找不到 NaN)console.log(weirdNumbers.includes(NaN)); // true (includes 能找到 NaN)

这个细节在处理一些可能包含非数字数据或计算结果为 NaN 的数组时,显得尤为重要。我个人就遇到过因为 indexOf 无法识别 NaN 而导致逻辑出错的情况,后来改用 includes 就迎刃而解了。所以,如果你的目标只是判断元素是否存在,并且不关心它的具体位置,那么 includes() 绝对是更现代、更健壮的选择。

includes() 如何处理复杂数据类型(对象、数组)的相等性?

这是 includes() 使用中一个常见的“陷阱”,或者说,是 JavaScript 相等性判断的固有特性。includes() 在进行元素比较时,使用的是严格相等性比较(===。这意味着,对于基本数据类型(如数字、字符串、布尔值),它会按值进行比较。但对于复杂数据类型(如对象、数组),它比较的是引用,而不是它们的内容。

举个例子:

const users = [{ id: 1, name: 'Alice' }, { id: 2, name: 'Bob' }];const alice = { id: 1, name: 'Alice' };const bob = users[1]; // 引用了数组中的第二个对象console.log(users.includes(alice)); // false// 尽管 alice 对象的内容和 users[0] 一模一样,但它们是两个不同的对象实例,内存地址不同。console.log(users.includes(bob)); // true// 因为 bob 变量直接引用了 users 数组中的第二个对象,它们指向同一个内存地址。const nestedArrays = [[1, 2], [3, 4]];const targetArray = [1, 2];console.log(nestedArrays.includes(targetArray)); // false// 同样,[1, 2] 即使内容相同,但作为不同的数组实例,引用也不同。console.log(nestedArrays.includes(nestedArrays[0])); // true

所以,如果你想在数组中查找一个内容相同但引用不同的对象,includes() 是帮不上忙的。这种情况下,你需要采取其他策略,比如使用 Array.prototype.some() 方法,结合一个自定义的比较函数来遍历数组,手动比较每个对象的属性:

const products = [{ id: 'a1', name: 'Laptop' }, { id: 'b2', name: 'Mouse' }];const searchProduct = { id: 'a1', name: 'Laptop' };const foundProduct = products.some(product => product.id === searchProduct.id && product.name === searchProduct.name);console.log(foundProduct); // true

这其实是 JavaScript 语言本身的一个设计选择,理解了这一点,就能避免很多不必要的困惑。

在大型数组中频繁使用 includes() 会有性能问题吗?有哪些替代方案?

是的,在处理大型数组时,频繁地调用 includes() 确实可能带来性能上的考量。includes() 方法在底层实现上,需要遍历数组中的每一个元素,直到找到目标元素或者遍历完整个数组。这意味着它的时间复杂度是 O(n)(线性时间复杂度),其中 n 是数组的长度。

对于小到中等规模的数组(比如几百、几千个元素),这种性能开销通常可以忽略不计。但如果你的数组包含成千上万甚至数十万个元素,并且你需要频繁地对这个数组进行 includes() 操作,那么每次 O(n) 的遍历累积起来,就会显著影响程序的响应速度。

想象一下,一个电商网站,你有一个包含百万商品ID的数组,每次用户搜索商品时,你都用 includes() 去检查某个ID是否存在,这显然不是一个高效的做法。

在这种情况下,我们可以考虑使用更适合快速查找的数据结构,最常见的替代方案是使用 JavaScript 的 Set 对象。

使用 Set 的替代方案:

Set 是一种允许你存储任何唯一值的集合。它的一个显著优势是,查找一个元素是否存在于 Set 中(使用 Set.prototype.has() 方法)的时间复杂度平均是 O(1)(常数时间复杂度),这比数组的 O(n) 要快得多。

当然,将数组转换为 Set 本身也需要时间,这个转换过程的时间复杂度是 O(n)。所以,这种优化适用于以下场景:

查找操作远多于添加/删除操作:你只需要一次性将数组转换为 Set,然后进行多次查找。数组内容相对稳定:如果数组内容频繁变化,每次变化都重新构建 Set 可能反而更慢。

const largeArray = Array.from({ length: 100000 }, (_, i) => `item-${i}`);// 转换为 Set (一次性开销)const itemSet = new Set(largeArray);// 查找操作 (O(1) 平均时间复杂度)console.time('includes check');console.log(largeArray.includes('item-99999')); // 假设这是最坏情况,遍历整个数组console.timeEnd('includes check'); // 可能会显示几十毫秒甚至更久console.time('Set has check');console.log(itemSet.has('item-99999')); // 几乎瞬间完成console.timeEnd('Set has check'); // 通常是零点几毫秒console.time('another includes check');console.log(largeArray.includes('non-existent-item'));console.timeEnd('another includes check');console.time('another Set has check');console.log(itemSet.has('non-existent-item'));console.timeEnd('another Set has check');

从上面的例子可以看到,对于大型数据集,Set.has() 在查找性能上有着压倒性的优势。因此,在设计需要高性能查找的系统时,考虑将数据存储在 Set 中,而不是单纯依赖数组的 includes(),是一个非常值得深思的优化方向。

以上就是JavaScript的Array.prototype.includes方法是什么?怎么用?的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1510321.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 05:54:04
下一篇 2025年12月20日 05:54:29

相关推荐

  • Node.js ES Modules中openai导入异常及误导性错误排查

    本文探讨了在Node.js ES Modules (ESM) 环境下使用openai npm包时,遇到的一个看似是SyntaxError的模块导入问题。文章详细分析了问题现象,揭示了其背后实则是一个与导入语句无关的运行时逻辑错误,并解释了为何这类深层问题可能导致误导性的编译或模块加载错误。通过代码示…

    2025年12月20日
    000
  • Node.js ES Modules与openai库的导入疑难解析

    本文深入探讨了在Node.js ES模块环境中,使用openai npm包时遇到的一个离奇的导入错误。尽管导入语句看似正确,系统却报告SyntaxError: The requested module ‘openai’ does not provide an export n…

    2025年12月20日
    000
  • 深入解析Node.js中误导性模块导入错误的排查与解决方案

    本文深入探讨了Node.js项目中一个看似是模块导入错误(SyntaxError: The requested module ‘openai’ does not provide an export named ‘Configuration’),但实际根…

    2025年12月20日
    000
  • 利用JavaScript清理动态内容:高效移除HTML Span元素中的括号

    本文旨在提供一个实用的JavaScript解决方案,用于处理第三方插件或动态内容生成中,HTML元素(如)自动添加多余字符(如括号)的问题。我们将重点介绍如何利用原生JavaScript遍历DOM并高效移除特定CSS类名元素中的括号,确保页面内容的整洁与准确呈现。此方法适用于客户端渲染内容,是解决此…

    2025年12月20日
    000
  • JavaScript await 行为深度解析:同步异常与异步流程控制

    本文深入探讨了 JavaScript 中 async/await 的核心机制,特别聚焦于 await 对同步抛出异常和异步返回值的不同处理方式。我们将解析 await 如何将操作数转换为 Promise,以及当非 async 函数在 Promise 创建前抛出错误时,为何 await 不会暂停执行,…

    2025年12月20日
    000
  • Promise.catch的错误捕获实践

    promise.catch 能捕获 promise 链中任何环节的拒绝及同步错误,但无法捕获链外同步错误、未包装成 promise 的异步错误及未处理的全局拒绝。1. 链外同步错误如 referenceerror 不在 promise 内部抛出则无法被捕获;2. settimeout 等独立异步操作…

    2025年12月20日 好文分享
    000
  • JavaScript中const和let如何替代var

    const声明对象后属性可以修改,因为const保证的是变量指向的内存地址不变,而非对象内部数据不可变。1. 对于基本数据类型,const确实防止值的修改;2. 对于对象类型,变量存储的是引用地址,修改对象属性不影响引用地址;3. 若要完全禁止对象修改,需使用object.freeze()方法。 在…

    2025年12月20日 好文分享
    000
  • JavaScript的JSON.parse和JSON.stringify怎么用?

    json.parse 用于将 json 字符串转换为 javascript 对象,而 json.stringify 则用于将 javascript 对象转换为 json 字符串。1. json.parse 通过反序列化接收 json 字符串并返回对象,若格式错误则抛出 syntaxerror;2. …

    2025年12月20日 好文分享
    000
  • 深入理解与实践:使用Jest测试Node.js REST GET请求封装函数

    本文详细介绍了如何使用Jest框架为Node.js中封装的REST GET请求函数编写单元测试。我们将深入探讨如何模拟HTTP请求(如https.get),处理异步回调,以及验证不同响应场景(成功、错误、JSON/非JSON数据)下的函数行为。通过具体的代码示例,帮助读者掌握高效、可靠的Node.j…

    2025年12月20日
    000
  • async函数中的并发执行控制

    并发控制在async函数中的核心目的是避免资源耗尽、接口限流或服务崩溃,通过限制同时运行的异步任务数量来维持系统稳定。1. 基于计数器和队列的自定义实现通过维护任务队列和执行计数器动态管理任务执行;2. 使用promise.all结合分块处理适用于固定任务列表的场景,将任务分为小批次串行执行;3. …

    2025年12月20日 好文分享
    000
  • ES6的Error子类如何自定义错误类型

    自定义es6错误类型能提升代码质量与错误处理的精确性。通过继承error类,开发者可创建具有语义化名称和附加上下文信息的错误类型,如validationerror和networkerror,从而告别模糊的错误提示。使用class语法定义错误类型时,需在构造函数中调用super()并设置name属性,…

    2025年12月20日 好文分享
    000
  • 如何处理JavaScript中的异步错误

    javascript中处理异步错误的核心方法包括使用async/await结合try/catch、promise的.catch()方法、promise.allsettled()以及全局错误监听机制。1. async/await与try/catch结合能以同步方式捕获异步错误,适用于现代异步编程;2.…

    2025年12月20日 好文分享
    000
  • JavaScript中异步编程的历史演变

    javascript异步编程的核心问题是单线程环境下高效处理耗时操作而不阻塞主线程。1. 最初使用回调函数,导致“回调地狱”,代码可读性和维护性差;2. promise引入状态管理和链式调用,解决了嵌套问题并统一了错误处理;3. async/await作为promise的语法糖,让异步代码几乎像同步…

    2025年12月20日 好文分享
    000
  • JavaScript中生成器与异步编程

    生成器在异步控制流中的核心作用是作为“流程协调员”,提供非阻塞式的暂停与恢复机制。①通过function*和yield关键字,允许函数中途暂停并将值“吐”出,外部通过next()方法传回值并继续执行;②支持以同步方式编写异步代码,提升可读性和维护性;③提供统一的错误处理机制,通过generator.…

    2025年12月20日 好文分享
    000
  • ES6中如何用数组的flatMap方法映射并展平

    flatmap为何出现?1.解决映射后展平常见场景,避免map+flat两步操作;2.提升代码可读性与意图表达清晰度;3.潜在性能优化,减少中间数组生成。flatmap是map后接flat(1)的语法糖,对每个元素应用回调并展平一层,使代码更简洁高效。例如,插入分隔项或提取多标签时,flatmap能…

    2025年12月20日 好文分享
    000
  • JavaScript如何用数组的at方法获取末尾元素

    在javascript中,数组的at()方法通过负数索引(如-1)更直观地获取末尾元素。传统方式需使用arr[arr.length – 1]进行计算,而at(-1)直接表达“获取最后一个元素”的意图,提升可读性;它支持链式调用,适用于复杂表达式、倒数任意元素获取、函数式编程风格及处理空数…

    2025年12月20日 好文分享
    000
  • Promise中的then方法详解

    then方法通过返回新promise实现链式调用,允许异步操作按顺序执行。1. then接受onfulfilled和onrejected回调,分别处理成功与失败;2. 回调执行结果决定新promise状态:返回值解决、抛出错误拒绝、返回promise则采纳其状态;3. 回调异步执行,确保一致性;4.…

    2025年12月20日 好文分享
    000
  • ES6的导出别名如何重命名模块

    es6中重命名模块导出通过as关键字实现,允许在不改变原始变量名的情况下以不同名字暴露。1. 重命名具名导出:使用export { originalname as newname }语法,如export { add as sum, subtract as minus }; 2. 重命名默认导出:通过…

    2025年12月20日 好文分享
    000
  • ES6中如何用正则表达式的d标志获取索引

    es6引入的正则d标志能获取捕获组索引。1. 使用d标志后,exec()返回的匹配对象新增indices属性;2. indices数组包含每个捕获组的[startindex, endindex]对;3. 该功能解决了手动计算索引易出错的问题;4. 可用于语法高亮、模板解析、富文本编辑等场景;5. m…

    2025年12月20日 好文分享
    000
  • JavaScript的this关键字是什么?如何正确使用?

    javascript中的this指向函数执行时的上下文,1. 默认绑定中,非严格模式下this指向全局对象,严格模式下为undefined;2. 隐式绑定中,this指向调用方法的对象;3. 显式绑定通过call、apply或bind指定this值;4. new绑定将this绑定到新创建的对象;5.…

    2025年12月20日 好文分享
    000

发表回复

登录后才能评论
关注微信