call和apply的使用-基础篇_build call apply,2024Web前端开发社招面试解答之性能优化

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Web前端全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上前端开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024c (备注前端)
img

正文

因为obox是一个元素对象,给obox元素添加点击事件,相当于给obox元素对象添加一个onclick属性,属性值为function,函数内部有一个this,当点击obox触发onclick,执行function时,打印出当前对象obox,依然符合this指向调用当前函数对象的原则。

综上所述,this的指向为:谁掉用当前this所在的函数,this就指向谁。也就是说,当前调用函数的那个对象自身就是this,就是当前的执行上下文。

三、执行上下文(this)的改变

执行上下文(this)是可以被改变的,为什么要改变执行上下文(this)呢,我们来模拟一种场景:

宿舍中,小A有每天洗头的习惯,每次洗完之后,头发湿漉漉的不方便,于是就攒钱买了一个吹风机,洗完之后吹一吹,神清气爽。小B洗头没有小A频繁,偶尔洗一次,洗完之后也是湿漉漉的不方便,但是又因为自己洗的次数少,所以不想再单独买一个吹风机,于是每次就借用小A的吹风机。

那么此时,我们如果把小A和小B都理解成一个对象,吹风机就是小A方法,它的所有人就是小A,小A在使用吹风机的时候,小A就是吹风就的执行上下文(this)。小B偶尔会需要用到吹风机,因为使用频次少,没必要重新买一个造成资源浪费,所以每次都是借用小A的,那么小B在使用吹风机的时候,吹风机被小B调用,此时小B就是吹风机的执行上下文(this)。此时吹风机的执行上下文(this)就被修改了。

(此情景仅为模拟,不适用于现实生活,现实生活中不推荐小B的做法)

在代码中,当一个对象A具有一个方法fn,另一个对象B没有方法,但是需要用到同样功能的fn方法时,可以通过改变A对象中函数fn的执行上下文(this)来实现调用,达到节约代码空间,不产生冗余函数的目的。如代码3-1(字面量创建对象写法):

var A = {
name: “AAA”,
fn: function(skill){
this.skill = skill;
console.log(“my name is " + this.name +”, my skills are " + this.skill);
}
}
var B = {
name: “BBB”
}
A.fn(“sing”); //my name is AAA, my skills are sing
B.fn(“dance”); //Uncaught TypeError: B.fn is not a function;

代码3-2(构造函数创建对象写法):

function ProA(name, skill){
this.name = name;
this.skill = skill;
this.fn = function(){
console.log(“my name is " + this.name +”, my skills are " + this.skill);
}
}
function ProB(name, skill){ }

var A = new ProA(“AAA”,“sing”);
A.fn(); //AAA

var B = new ProB(“BBB”,“dance”);
B.fn(); //Uncaught TypeError: B.fn is not a function;

那么当我们确定好需求之后,接下来的操作就简单了,只要能找到一种方法,能够将对象A中函数fn的上下文修改成B对象,就可以解决这些问题。

此时,就可以使用call和apply这两个函数的方法,接下来我们只需要如何使用call和apply即可。

四、call和apply的使用

以上可得知call和apply这两个方法的功能是:用来修改函数的执行上下文(this)。

call和apply其实都是函数的方法,我们知道方法是对象中的函数,那么函数怎么还可以有函数呢,我们可以结合js中万物皆对象这句话,其实function在js中也是一个对象(可结合对象的原型来理解了,此处暂不做深究,了解原型请参考下篇文档),所以函数也有方法。

那么这两个方法如何使用呢,我们先来看完整语法:

call(thisObj,arg1,arg2,arg3,……)

call方法接收一个或一个以上的参数,当接收一个参数时,第一个参数表示要改变的原函数的执行上下文(this);接收多个参数时,第二个参数及后面所有参数用来替换原函数的参数。

将代码3-1使用call方法改成如下代码4-1方式,即可让对象B具有对象A的fn方法,代码4-1:

var A = {
name: “AAA”,
fn: function(skill){
this.skill = skill;
console.log(“my name is " + this.name +”, my skills are " + this.skill);
}
}
var B = {
name: “BBB”
}
A.fn(“sing”); //my name is AAA, my skills are sing
//此处改动产生的效果为:
//在执行A对象的函数fn时,通过call将函数fn的执行上下文(this)暂时修改为对象B,
//此时fn中的this指向对象B,同时修改原函数fn的参数为“dance”,
//call方法自动执行改变之后的原函数
A.fn.call(B,“dance”); //my name is BBB, my skills are dance

将代码3-2使用call方法改成如下代码4-2方式,构造函数方式创建对象写法,代码4-2

function ProA(name, skill){
this.name = name;
this.skill = skill;
this.fn = function(){
console.log(“my name is " + this.name +”, my skills are " + this.skill);
}
}
function ProB(name, skill){
//此处改动产生的效果为:
//在ProB内,通过apply,执行,并改动ProA中的执行上下文(this),
//及修改ProA的参数为ProB所接收的参数
//那么在new调用ProB时,相当于调用了被修改了执行上下文和参数之后的ProA
ProA.call(this, name, skill)
}
var A = new ProA(“AAA”,“sing”);
A.fn(); //my name is AAA, my skills are sing
var B = new ProB(“BBB”,“dance”);
B.fn(); //my name is BBB, my skills are dance

apply(thisObj,argArr)

apply方法接受一个或两个参数,当接收一个参数时,第一个参数表示要改变的原函数的执行上下文(this);接收两个参数时,第二个参数必须是数组(或伪数组),用于替换原函数中arguments保存的参数,

将代码3-1使用apply方法改成如下代码4-3方式,即可让对象B具有对象A的fn方法,代码4-3:

var A = {
name: “AAA”,
fn: function(skill){
this.skill = skill;
console.log(“my name is " + this.name +”, my skills are " + this.skill);
}
}
var B = {
name: “BBB”
}
A.fn(“sing”); //my name is AAA, my skills are sing
//此处改动产生的效果为:
//在执行A对象的函数fn时,通过apply将函数fn的执行上下文(this)暂时修改为对象B,
//此时fn中的this指向对象B,同时修改原函数fn的参数为“dance”(注意“dance”参数必须是数组的形式),
//apply方法自动执行改变之后的原函数
A.fn.apply(B,[“dance”]); //my name is BBB, my skills are dance

将代码3-2使用call方法改成如下代码4-2方式,构造函数方式创建对象写法

function ProA(name, skill){
this.name = name;
this.skill = skill;
this.fn = function(){
console.log(“my name is " + this.name +”, my skills are " + this.skill);
}
}
function ProB(name, skill){
//此处改动产生的效果为:
//在ProB内,通过apply,执行,并改动ProA中的执行上下文(this),
//及修改ProA的参数为ProB所接收的参数(注意:此时的参数必须是一个数组的格式)
//那么在new调用ProB时,相当于调用了被修改了执行上下文和参数之后的ProA
ProA.apply(this, [name, skill])
//参数也可以写成arguments的形式,arguments属于伪数组,但是也可以被apply所接收处理,如:
//ProA.apply(this, arguments)
}
var A = new ProA(“AAA”,“sing”);

总结

阿里十分注重你对源码的理解,对你所学,所用东西的理解,对项目的理解。

最新阿里蚂蚁金服四面(已拿offer)Java技术面经总结

最新阿里蚂蚁金服四面(已拿offer)Java技术面经总结

最新阿里蚂蚁金服四面(已拿offer)Java技术面经总结

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024c (备注前端)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

需要这份系统化的资料的朋友,可以添加V获取:vip1024c (备注前端)
[外链图片转存中…(img-zLenJLR4-1713652680824)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值