原贴
今天闲来无事重写了这个方法.大概意思就是根据属性名
判断是否是post开头,如果是的话将这个属性对应的方法进行改写,先进行一个随机数判断
再决定是否执行原方法。正确的写法是用匿名函数
或者let
之前的帖子中如果我舍弃用匿名函数的话(也就是错误写法),只有post3
这个属性会被改写,因为i
和func
的指向都是最后一次for中i的值,所以被覆盖了。但是这一次我的代码中3个方法都被改写成了post3的方法,obj[i]并没有被覆盖。让我觉得非常疑惑。为什么两次相同的代码结果会不一样。
var obj = {
post1: function() {
console.log('post1')
},
post2: function() {
console.log('post2')
},
post3: function() {
console.log('post3')
},
check: function() {
return Math.random() > 0.5
}
}
function changeFunc(obj){
for(var i in obj){
if (i.indexOf('post')==0 && typeof obj[i] == 'function') {
var func = obj[i]
obj[i] = function() {
if (obj.check()) {
func.apply(this, arguments)
}
}
}
}
}
changeFunc(obj)
obj.post1()
obj.post2()
obj.post3()
结果是随机输出0-3个'post3',这是为啥呢= =
闭包与变量提升问题。
上面代码中变量
func
使用var
定义的,会被提升到函数开头,所以你重写的三个函数都是公用同一个变量func
,调用的肯定也是同一个函数,即遍历中最后满足条件的函数。将
var
改为let
就没问题了,let
定义的变量不会提升。