标签:语句 btn return turn 不难 语法规则 doc The 函数参数
最近面试经常碰到这样一道题(或者类似):ps.因为我阿姨不会就详细分析吧。
var btns = document.getElementsByClassName('btn');
for(let i = 0; i < btns.length; i++){
btn[i].onclick = function(){
console.log(i)
}
}
这个之前说过,因为面试的时候,说了主要看闭包。所以也没有多想。
function fun(){
for (var i = 0; i < 5; i++){
setTimeout(()=>{
console.log(i)
},0);
}
}
fun();
其实,通过分析就不难看出,核心点问题都是, 在一个循环内,延时打印循环变量。 所有的结果一样。
因为js的变量有提升作用,所以可以将程序改成成更加直观的形式。
var i = 0;
function fun(){
while (i < 5){
setTimeout(()=>{
console.log(i) // 这里访问的i 都是 之前定义的 i
},1000);
i++; // 这里访问的i 也是 之前定义的 i
}
}
fun();
我们改成程序之后,就可以看到了。 打印语句的i,因为都是在等待i变换之后,取i值打印, 所以结果一样。
这里产生问题的原因在于:
其实,核心问题,就是等待循环结束,才去访问i,而且访问的是同一个i。
解决思路:两种 改变访问时机, 改变访问变量。
for( var i = 0; i < 5; i++){
console.log(i);
}
这样做好像没什么意义。
function fun(){
for (let i = 0; i < 5; i++){
setTimeout(()=>{
console.log(i)
},0);
}
}
fun();
用let代替var来声明变量,就可以把变量的作用域限制在当前代码块中也就是{}
function fun(){
for (var i = 0; i < 5; i++){
setTimeout((function(i_){
return function(){
console.log(i_)
}
})(i),0);
}
}
fun();
其实, 也就是用 立即执行函数参数i_ 来保存 i 值 。 当然你也可以把i_写成i。
分割线(这对之前问题的处理)
function fun(){
for (var i = 0; i < 5; i++){
var promise = new Promise(function(resolve, reject){
setTimeout(resolve,1000,i);
}); // 这个是立即执行
promise.then(function(value){ // 回调执行
console.log(value);
})
}
}
fun();
下面代码也可以实现,其实,你就会发现, 这是因为 setTimeout 函数 可以调用给函数传递参数的原因, 。
function fun(){
for (var i = 0; i < 5; i++){
setTimeout((i)=>{
console.log(i)
},0,i);
}
}
fun();
标签:语句 btn return turn 不难 语法规则 doc The 函数参数
原文地址:https://www.cnblogs.com/cyrus-br/p/10519523.html