1.apply赶掖、call 有什么作用露筒,什么區(qū)別
函數(shù)調(diào)用有三種形式:
func(p1,p2)
obj.childrend.method(p1,p2)
func.call(context,p1,p2)
第三種形式才是正常調(diào)用形式,call就是調(diào)用函數(shù)的一種形式甸鸟,call的第一個參數(shù)(context)很重要,context就是函數(shù)執(zhí)行的上下文環(huán)境,也就是this。call一個函數(shù)時傳入的context是什么励稳,函數(shù)的this就指向什么。如果傳入的context是null或者undefined囱井,那么context默認(rèn)就是window(嚴(yán)格模式下默認(rèn) context 是 undefined)
apply也是調(diào)用函數(shù)的一種形式驹尼,和call不同之處在于apply調(diào)用是只能傳入兩個參數(shù).apply(context,參數(shù)數(shù)組),.call(context,p1,p2)可傳入多個參數(shù)庞呕。
2.下面代碼輸出什么新翎,為什么
func()
function func() {
alert(this)
}
func()等價于func.call()
輸出的內(nèi)容是window全局對象
3.以下代碼輸出什么?
var john = {
firstName: "John"
}
function func() {
alert(this.firstName + ": hi!")
}
john.sayHi = func
john.sayHi()
john.sayHi()可寫成john.sayHi.call(john),這里call傳入的context是john這個對象,所以this.firstNme的值是'John',打印結(jié)果是John :hi
4.下面代碼輸出什么
function fn0(){
function fn(){
console.log(this);
}
fn();
}
fn0();
document.addEventListener('click', function(e){
console.log(this);
setTimeout(function(){
console.log(this);
}, 200);
}, false);
fn0()執(zhí)行后打印的this是全局對象window住练,因?yàn)閒n0()執(zhí)行的時候是window在調(diào)用地啰。
事件綁定后第一次打印的this是觸發(fā)該事件的dom對象,第二次打印的this是window全局對象讲逛,因?yàn)閟etTimeout和setInterval執(zhí)行時也是window全局對象在調(diào)用髓绽。
5.下面代碼輸出什么,為什么
var john = {
firstName: "John"
}
function func() {
alert( this.firstName )
}
func.call(john)
.call(context)這里context代表傳入的函數(shù)執(zhí)行的上下文環(huán)境妆绞,這里傳入的是john對象顺呕,所以func.call(john)打印的內(nèi)容是John
6.代碼輸出?
var john = {
firstName: "John",
surname: "Smith"
}
function func(a, b) {
alert( this[a] + ' ' + this[b] )
}
func.call(john, 'firstName', 'surname')
func.call(context,參數(shù)1,參數(shù)2) 這里函數(shù)執(zhí)行時傳入的this是john對象括饶,所以打印的內(nèi)容是John Smith
7.以下代碼有什么問題株茶,如何修改
var module= {
bind: function(){
$btn.on('click', function(){
console.log(this) //this指什么
this.showMsg();
})
},
showMsg: function(){
console.log('饑人谷');
}
}
console.log(this)打印的內(nèi)容是觸發(fā)綁定事件的DOM對象,這里是$btn觸發(fā)所以this指向$btn图焰。代碼中執(zhí)行this.showMsg()會報錯启盛,因?yàn)閠his指向$btn后,$btn上沒有.showMsg()這個方法,.showMsg()方法在module對象上僵闯。修改代碼如下
var module= {
bind: function(){
var cur=this//申明cur卧抗,將this賦值給cur,這里this指向module對象
$btn.on('click', function(){
console.log(this) //綁定事件后this在這里指向$btn
cur.showMsg();//cur指向module對象
})
},
showMsg: function(){
console.log('饑人谷');
}
}
8.下面代碼輸出什么
var length = 3;
function fa() {
console.log(this.length);
}
var obj = {
length: 2,
doSome: function (fn) {
fn();
arguments[0]();
}
}
obj.doSome(fa)
obj.doSome(fa)等價于
obj.doSome.call(obj,fa)鳖粟,注意雖然這里this指向obj社裆,但參數(shù)2位置上是fa函數(shù),doSome定義傳入?yún)?shù)并執(zhí)行'fn()'向图,所以參數(shù)2位置上的fa()要執(zhí)行泳秀,fa()等價于fa.call(null),因?yàn)閚ull所以this指向全局對象window,因?yàn)閘ength=3實(shí)在全局對象下聲明的,所以第一次打印結(jié)果為3.
執(zhí)行arguments[0]()
時榄攀,等價于arguments[0].call(arguments)由于arguments是類數(shù)組對象嗜傅,相當(dāng)于傳入的第0個參數(shù)執(zhí)行,這里傳入的第0個參數(shù)是fa,所以等價于fa.call(arguments),這時候this指向arguments類數(shù)組對象檩赢,arguments.length=1吕嘀,因?yàn)橹挥?個參數(shù)fa(),所以第二次打印1
9.下面代碼輸出什么? why
obj = {
go: function() { alert(this) }
}
obj.go();
(obj.go)();
(a = obj.go)();
(0 || obj.go)();
1.obj.go()等價于obj.go.call(obj) this指向obj贞瞒,所以打印obj對象
2.(obj.go)()相當(dāng)于obj對象上的go屬性立即執(zhí)行币他,go是個函數(shù)所以等價于obj.go.call(obj)打印obj對象
3.obj.go賦值給全局對象a,等價于a=function(){alert(this)}憔狞,a()---->a.call(),這里打印的是window全局對象
4.注意這里執(zhí)行或運(yùn)算函數(shù)時蝴悉,是全局對象window在調(diào)用,因?yàn)榈谝粋€0不成立所以等價于obj.go.call()瘾敢。所以打印window全局對象
****本文版權(quán)歸本人和饑人谷所有拍冠,轉(zhuǎn)載請注明來源。****