为什么这段 JavaScript 代码中的 `i` 始终输出 6?

为什么这段 javascript 代码中的 `i` 始终输出 6?

js 闭包问题

给定以下代码:

                document            ul li:nth-child(1) {            background: #00ffff;        }        ul li:nth-child(2) {            background: #0011ff;        }        ul li:nth-child(3) {            background: #ff00ff;        }        ul li:nth-child(4) {            background: #aaffff;        }        ul li:nth-child(5) {            background: #ffff00;        }        ul li:nth-child(6) {            background: #00ff00;        }                                
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
var oinput = document.getelementsbytagname("input"); var oli = document.getelementsbytagname("li"); for (var i = 0; i < oinput.length; i++) { oinput[i].index = i; oinput[i].onclick = function() { for (var j = 0; j < oli.length; j++) { console.log(i); // i输出为6 oli[j].style.display = "block"; } console.log(i); // i输出为6 oli[this.index].style.display = "none"; } }

问题:

执行此代码时,为什么输出的始终是 6?

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

答案:

这是由于 javascript 中的闭包所致。当为每个输入按钮设置单击事件处理程序时,会创建一个闭包,该闭包包含当前的 i 值。单击任何按钮时,都会调用闭包,并将当前的 i 值打印到控制台。然而,由于 i 是闭包内的变量,当单击最后一个按钮时,它始终保留其最终值,即 6。

要解决此问题,可以在单击事件处理程序内使用立即调用函数表达式 (iife) 来创建新作用域,并为每个按钮指定一个唯一的 i 值:

for (var i = 0; i < oInput.length; i++) {  oInput[i].index = i;  (function(index) {    oInput[i].onclick = function() {      for (var j = 0; j < oLi.length; j++) {        console.log(index);        oLi[j].style.display = "block";      }      console.log(index);      oLi[index].style.display = "none";    }  })(i);}

以上就是为什么这段 JavaScript 代码中的 `i` 始终输出 6?的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 17:56:04
下一篇 2025年12月19日 17:56:13

相关推荐

  • js如何判断变量是否未定义 未定义检测的5种实用技巧

    在javascript中判断变量是否未定义的核心方法是使用typeof操作符。1. typeof是最常用且安全的方法,不会因变量未定义而报错,直接返回”undefined”;2. 可结合window对象的属性判断全局变量是否存在,如使用in操作符或hasownproperty…

    2025年12月20日 好文分享
    000
  • js作用域scope链解析_js作用域scope链详细说明

    javascript的作用域链是变量查找的机制,决定了变量的可访问性。1. 引擎首先在当前作用域查找变量,若未找到则沿作用域链向上查找,直到全局作用域;2. 作用域链由词法作用域决定,函数定义时确定,支撑闭包的实现;3. 闭包通过作用域链访问外部函数的变量,即使外部函数已执行完毕;4. 避免问题需限…

    2025年12月20日 好文分享
    000
  • js闭包closure原理是什么_js闭包closure深度解析

    闭包是函数与其词法环境的绑定,允许函数访问外部变量。1. 闭包解决数据封装和状态保持问题;2. 通过隐藏变量实现私有性,保持函数执行后状态;3. 应用于事件处理、模块化和柯里化;4. 闭包会延长变量生命周期,需手动解除引用以避免内存泄漏;5. 闭包不影响this指向,但可通过闭包间接访问外部this…

    2025年12月20日 好文分享
    000
  • js如何实现模块化加载 模块化加载的5种方案对比

    在javascript中实现模块化加载的核心在于解决代码组织、依赖管理和命名冲突的问题,常用方案包括iife、commonjs、amd、umd和es modules。①iife通过函数作用域封装变量和函数,适合简单项目但易全局污染;②commonjs适用于node.js环境,同步加载需打包工具支持;…

    2025年12月20日 好文分享
    000
  • JS中的闭包是什么?如何实现?

    闭包是指函数能够访问并记住其词法作用域,即使在其作用域外执行。1. 闭包通过嵌套函数引用外部函数变量实现;2. 常见实现方式包括函数返回函数或将函数作为参数传递;3. 实际用途有封装私有变量、数据缓存、柯里化函数和事件回调;4. 使用时需注意内存占用、调试困难和性能影响等问题,应合理控制生命周期以避…

    2025年12月20日
    000
  • JS中的箭头函数和普通函数有什么区别?

    箭头函数与普通函数的关键区别在于this绑定、构造函数能力和arguments对象。1. this指向不同:普通函数的this取决于调用方式,而箭头函数继承外层作用域的this,如在对象方法中使用可能无法访问对象属性;2. 箭头函数不能作为构造函数,无法通过new创建实例;3. 箭头函数无自己的ar…

    2025年12月20日
    000
  • JS中的let和var有什么区别?怎么用?

    在 javascript 中,let 和 var 的主要区别在于作用域、变量提升和重复声明。1. let 是块级作用域,而 var 是函数作用域;例如,在 if 块中用 let 声明的变量无法在外部访问,var 则可以。2. var 存在变量提升,即变量可在声明前访问但值为 undefined,而 …

    2025年12月20日
    000
  • JS中的默认参数怎么用?有什么作用?

    javascript 中的默认参数用于在函数调用时未传参或参数为 undefined 时提供替代值。其核心作用是提升函数容错能力,避免意外错误。基本写法是在定义参数时赋默认值,如 function greet(name = “guest”)。只有参数为 undefined 时…

    2025年12月20日
    000
  • JS中的this指向什么?怎么控制?

    this的指向取决于函数调用方式。1. 默认情况下,普通函数的this指向全局对象(如浏览器中为window),若作为对象方法调用则指向该对象,若通过new调用则指向新对象实例;2. 箭头函数无自身this,继承外层作用域的this;3. 可使用call、apply或bind手动绑定this,其中c…

    2025年12月20日
    000
  • 如何在JavaScript中定义函数?

    在javascript中,可以通过函数声明、函数表达式、箭头函数和function构造函数四种方式定义函数。1.函数声明(function greet(name) { return hello, ${name}!; })直观且会提升。2.函数表达式(const greet = function(na…

    2025年12月20日
    000
  • 什么是JavaScript中的模块化?

    javascript中的模块化是将代码组织成独立的、可复用的模块,每个模块负责特定功能,提高代码的可维护性和可扩展性。模块化的发展经历了iife、commonjs、amd,到现在的es6模块。使用es6模块时需要注意:1. 模块的加载顺序和依赖管理,2. 模块的性能优化,3. 模块的测试和调试。 什…

    2025年12月20日
    000
  • JavaScript中的let和var有什么区别?

    let和var的主要区别在于作用域和变量提升:1. let遵循块级作用域,不会变量提升;2. var遵循函数作用域,会变量提升。使用let可以提高代码的可读性和可维护性,减少错误。 让我们探讨一下JavaScript中的let和var有什么区别,这个问题在开发过程中经常被提及,尤其是在处理变量作用域…

    2025年12月20日
    000
  • JavaScript中的this关键字指向什么?

    在javascript中,this的指向取决于函数的调用方式。1)全局环境中,this指向全局对象;2)作为对象方法调用时,this指向该对象;3)从对象中提取方法调用时,this可能指向全局对象;4)使用箭头函数或bind方法可以固定this的指向;5)箭头函数没有自己的this,适合处理回调函数…

    2025年12月20日
    000
  • 怎样用JavaScript创建单例?

    在javascript中,单例模式可以通过闭包或es6类语法实现。1)闭包方法使用自执行函数和getinstance方法管理实例。2)es6类语法使用静态方法getinstance管理实例。使用单例模式时需注意全局状态管理、性能和测试难度,并遵循避免滥用、考虑替代方案和模块化设计的最佳实践。 在Ja…

    2025年12月20日
    000
  • 什么是JavaScript中的闭包?

    闭包是javascript中允许函数访问外部作用域变量的特性。1)闭包通过捕获词法环境实现,即使外部函数执行完毕,变量仍可访问。2)闭包应用于私有变量、模块模式和事件处理。3)注意闭包可能导致内存泄漏和代码复杂性,需谨慎使用并确保代码可读性。 闭包是JavaScript中一个非常强大的特性,但也常常…

    2025年12月20日
    000
  • JavaScript中的bind方法有什么作用?

    javascript中的bind方法用于创建一个新的函数,其this值被永久绑定到bind方法的参数上。1)bind方法可以确保函数的this上下文不变,适用于回调函数和事件处理。2)使用bind时需注意性能和内存问题,因为每次调用会创建新函数。3)箭头函数可替代bind,避免内存泄漏,因为其thi…

    2025年12月20日
    000
  • JavaScript中如何避免内存泄漏?

    避免javascript内存泄漏的方法包括:1. 清除定时器和回调函数,2. 谨慎使用闭包并手动释放大对象,3. 及时释放dom引用,4. 移除不再需要的事件监听器。通过这些措施和使用性能优化工具,可以有效减少内存泄漏,提升应用性能。 哦,JavaScript中的内存泄漏问题,这可是每个开发者都头疼…

    2025年12月20日
    000
  • 如何用JavaScript创建一个简单的函数?

    javascript函数的创建和使用可以通过以下步骤实现:基本函数创建:使用function greet(name) { return hello, ${name}!; }创建简单函数。函数作为第一等公民:函数可以被传递、赋值和调用,支持复杂逻辑如闭包和高阶函数。函数设计:注意命名、参数和复杂度,保…

    2025年12月20日
    000
  • JavaScript中如何删除Cookie?

    在javascript中删除cookie的方法是设置其过期时间为过去的时间。具体步骤包括:1. 使用deletecookie函数,将cookie的过期时间设置为1970年1月1日,并确保路径一致;2. 如果cookie是在子域名下设置的,需要在删除时指定相同的域名;3. 注意secure和httpo…

    2025年12月20日
    000
  • JavaScript中如何使用setTimeout()?

    在javascript中使用settimeout()的步骤如下:1. 基本用法:settimeout(function() { console.log(‘hello, world!’);}, 1000)会在1秒后执行。2. 传递参数:使用箭头函数,如settimeout(()…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信