我的编程空间,编程开发者的网络收藏夹
学习永远不晚

编程语言中this、call、apply的用法分析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

编程语言中this、call、apply的用法分析

小编给大家分享一下编程语言中this、call、apply的用法分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!

前言+思考题

记得当时找实习的时候,总是会在简历上加上一句——熟悉Js,例如this指向、call、apply等…

而每次投递简历时我都会经历如下步骤

  • 面试前,去问度娘——this指向可以分为哪几种啊~、call和apply的区别是什么?底气由0% 猛涨到了 50%;

  • 面试中,面试官随便扔上来几道题,我都可以“坚定的”给出答案,结果总是不尽人意…

  • 面试后,我会羞愧的删除掉简历上的这一条。而再之后投递简历时我又再次加上了这一条…

思考题

下面几道题是我在网上搜索出来的热度较高的问题,如果大佬们可以轻松的回答上,并有清晰的思路,不妨直接点个赞吧(毕竟也消耗了不少脑细胞),如果大佬们能在评论处指点一二,就更好了!!!

填空题:

  • 执行Javascript中的【 】函数会创建一个新函数,新函数与被调函数具有相同的函数体,当目标函数被调用时 this 值指向第一个参数。

问答题:

  • 请你谈一下改变函数内部this指针的指向函数有哪几种,他们的区别是什么?

  • this的指向可以分为哪几种?

代码分析题:

var name = 'window'var person1 = {  name: 'person1',  show1: function () {    console.log(this.name)  },  show2: () => console.log(this.name),  show3: function () {    return function () {      console.log(this.name)    }  },  show4: function () {    return () => console.log(this.name)  }}var person2 = { name: 'person2' }person1.show1()person1.show1.call(person2)person1.show2()person1.show2.call(person2)person1.show3()()person1.show3().call(person2)person1.show3.call(person2)()person1.show4()()person1.show4().call(person2)person1.show4.call(person2)()

一、this的指向

百度、谷歌上输入“this的指向”关键字,大几千条文章肯定是有的,总不至于为了全方面、无死角的掌握它就要将所有的文章都看一遍吧?所以不如梳理出一个稳固的框架,顺着我们的思路来填充它。

思维导图

编程语言中this、call、apply的用法分析

本节精华:
  • this 总是(非严格模式下)指向一个对象,而具体指向哪个对象是在运行时基于函数的执行环境动态绑定的,而非函数被声明时的环境;

  • 除了不常用的with和eval的情况,具体到实际应用中,this指向大概可以分为四种:

    • 作为对象的方法调用;

    • 作为普通函数调用;

    • 构造器调用;

    • call 或 apply调用;

    • 箭头函数中,this指向函数上层作用域的this;

  • 构造器普通函数的区别在于被调用的方式

  • A,call(B) => 可以理解成在B的作用域内调用了A方法;

分析

1、作为对象的方法调用

当函数作为对象的方法被调用时,this指向该对象

var obj = {    a: 'yuguang',    getName: function(){        console.log(this === obj);        console.log(this.a);    }};obj.getName(); // true yuguang

2、作为普通函数调用

当函数不作为对象的属性被调用,而是以普通函数的方式,this总是指向全局对象(在浏览器中,通常是Window对象)

window.name = 'yuguang';var getName = function(){    console.log(this.name);};getName(); // yuguang

或者下面这段迷惑性的代码:

window.name = '老王'var obj = {    name: 'yuguang',    getName: function(){        console.log(this.name);    }};var getNew = obj.getName;getNew(); // 老王

而在ES5的严格模式下,this被规定为不会指向全局对象,而是undefined

3、构造器调用

除了一些内置函数,大部分Js中的函数都可以成为构造器,它们与普通函数没什么不同

构造器普通函数的区别在于被调用的方式
当new运算符调用函数时,总是返回一个对象,this通常也指向这个对象

var MyClass = function(){    this.name = 'yuguang';}var obj = new MyClass();obj.name; // yuguang

但是,如果显式的返回了一个object对象,那么此次运算结果最终会返回这个对象。

var MyClass = function () {    this.name = 1;    return {        name: 2    }}var myClass = new MyClass(); console.log('myClass:', myClass); // { name: 2}

只要构造器不显示的返回任何数据,或者返回非对象类型的数据,就不会造成上述问题。

4、call或apply调用

跟普通的函数调用相比,用call和apply可以动态的改变函数的this

var obj1 = {    name: 1,    getName: function (num = '') {        return this.name + num;    }};var obj2 = {    name: 2,};// 可以理解成在 obj2的作用域下调用了 obj1.getName()函数console.log(obj1.getName()); // 1console.log(obj1.getName.call(obj2, 2)); // 2 + 2 = 4console.log(obj1.getName.apply(obj2, [2])); // 2 + 2 = 4

5.箭头函数

箭头函数不会创建自己的this,它只会从自己的作用域链的上一层继承this。

因此,在下面的代码中,传递给setInterval的函数内的this与封闭函数中的this值相同:

this.val = 2;var obj = {    val: 1,    getVal: () => {        console.log(this.val);    }}obj.getVal(); // 2

常见的坑

就像标题一样,有的时候this会指向undefined

情况一

var obj = {    name: '1',    getName: function (params) {        console.log(this.name)    }};obj.getName();var getName2 = obj.getName;getName2();

这个时候,getName2()作为普通函数被调用时,this指向全局对象——window。

情况二

当我们希望自己封装Dom方法,来精简代码时:

var getDomById = function (id) {    return document.getElementById(id);};getDomById('p1') //dom节点

那么我们看看这么写行不行?

var getDomById = document.getElementByIdgetDomById('p1') // Uncaught TypeError: Illegal invocation(非法调用)

这是因为:

  • 当我们去调用document对象的方法时,方法内的this指向document

  • 当我们用getId应用document内的方法,再以普通函数的方式调用,函数内容的this就指向了全局对象。

利用call和apply修正情况二

document.getElementById = (function (func) {    return function(){        return func.call(document, ...arguments)    }})(document.getElementById) // 利用立即执行函数将document保存在作用域中

二、call和apply

不要因为它的“强大”而对它产生抗拒,了解并熟悉它是我们必须要做的,共勉!

思维导图

编程语言中this、call、apply的用法分析

1.call和apply区别

先来看区别,是因为它们几乎没有区别,下文代码实例call和apply都可以轻易的切换。

当它们被设计出来时要做到的事情一摸一样,唯一的区别就在于传参的格式不一样

  • apply接受两个参数

    • 第一个参数指定了函数体内this对象的指向

    • 第二个参数为一个带下标的参数集合(可以是数组或者类数组)

  • call接受的参数不固定

    • 第一个参数指定了函数体内this对象的指向

    • 第二个参数及以后为函数调用的参数

因为在所有(非箭头)函数中都可以通过arguments对象在函数中引用函数的参数。此对象包含传递给函数的每个参数,它本身就是一个类数组,我们apply在实际使用中更常见一些。

call是包装在apply上面的语法糖,如果我们明确的知道参数数量,并且希望展示它们,可以使用call。

当使用call或者apply的时候,如果我们传入的第一个参数为null,函数体内的this会默认指向宿主对象,在浏览器中则是window

借用其他对象的方法

我们可以直接传null来代替任意对象

Math.max.apply(null, [1, 2, 3, 4, 5])
2.call和apply能做什么?

使用一个指定的 this 值和单独给出的一个或多个参数来调用一个函数——来时MDN

  • 调用构造函数来实现继承;

  • 调用函数并且指定上下文的 this;

  • 调用函数并且不指定第一个参数;

1.调用构造函数来实现继承

通过“借用”的方式来达到继承的效果:

function Product(name, price) {this.name = name;this.price = price;}function Food(name, price) {Product.call(this, name, price); //this.category = food;}var hotDog = new Food('hotDog', 20);

2.调用函数并且指定上下文的 this

此时this被指向了obj

function showName() {    console.log(this.id + ':' + this.name);};var obj = {    id: 1,    name: 'yuguang'};showName.call(obj)

3.使用call单纯的调用某个函数

Math.max.apply(null, [1,2,3,10,4,5]); // 10

三、模拟实现一个call

先来看一下call帮我们需要做什么?

var foo = {value: 1};function show() {console.log(this.value);};show.call(foo); //1

就像解方程,要在已知条件中寻找突破哦口:

  • call 使得this的指向变了,指向了foo;

  • show 函数被执行了;

  • 传入的参数应为 this + 参数列表;

第一版代码

上面提到的3点,仅仅完成了一点,且传入的参数

var foo = {    value: 1};function show() {    console.log(this.value);};Function.prototype.setCall = function (obj) {    console.log(this); // 此时this指向show    obj.func = this; // 将函数变成对象的内部属性    obj.func(obj.value); // 指定函数    delete obj.func // 删除函数,当做什么都没发生~}show.setCall(foo);

第二版代码

为了解决参数的问题,我们要能获取到参数,并且正确的传入:

var foo = {    value: 1};function show(a, b) {    console.log(this.value);    console.log(a + b);};Function.prototype.setCall = function (obj) {    obj.fn = this; // 将函数变成对象的内部属性    var args = [];    for(let i = 1; i < arguments.length; i++){        args.push('arguments[' + i + ']');    }    eval('obj.fn(' + args + ')'); // 传入参数    delete obj.fn; // 删除函数,当做什么都没发生~}show.setCall(foo, 1, 2); // 1 3

此时,我们就可以做到,传入多个参数的情况下使用call了,但是如果你仅想用某个方法呢?

第三版代码

Function.prototype.setCall = function (obj) {    var obj = obj || window;    obj.fn = this;    var args = [];    for(var i = 1, len = arguments.length; i < len; i++) {    args.push('arguments[' + i + ']');  }  var result = eval('obj.fn(' + args +')');  delete obj.fn;  return result;};// 测试一下var value = 2;var obj = { value: 1 };function bar(name, age) {  console.log(this.value);  return {    value: this.value,    name: name,    age: age  }}bar.setCall(null); // 2console.log(bar.setCall(obj, 'yuguang', 18));

四、bind

bind() 方法创建一个新的函数,在 bind() 被调用时,这个新函数的 this 被指定为 bind() 的第一个参数,而其余参数将作为新函数的参数,供调用时使用 —— MDN

提到了callapply,就绕不开bind。我们试着来模拟一个bind方法,以便加深我们的认识:

Function.prototype.bind = function (obj) {    var _this = this; // 保存调用bind的函数    var obj = obj || window; // 确定被指向的this,如果obj为空,执行作用域的this就需要顶上喽    return function(){        return _this.apply(obj, arguments); // 修正this的指向    }};var obj = {    name: 1,    getName: function(){        console.log(this.name)    }};var func = function(){    console.log(this.name);}.bind(obj);func(); // 1

这样看上去,返回一个原函数的拷贝,并拥有指定的 this 值,还是挺靠谱的哦~

以上是“编程语言中this、call、apply的用法分析”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注编程网行业资讯频道!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

编程语言中this、call、apply的用法分析

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

编程语言中this、call、apply的用法分析

小编给大家分享一下编程语言中this、call、apply的用法分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!前言+思考题记得当时找实习的时候,总是会在简历上
2023-06-14

简述Java编程语言中的逃逸分析

大家一般认为new出来的对象都是被分配在堆上,但这并不是完全正确,通过对Java对象分配过程分析,我们发现对象除了可以被分配在堆上,还可以在栈或TLAB中分配空间。而栈上分配对象的技术基础是逃逸分析和标量替换,本文主要介绍下
2023-05-31

Go语言与其他编程语言的比较分析

Go语言与其他编程语言的比较分析随着互联网的不断发展和计算机技术的日新月异,编程语言作为实现计算机程序的工具也在不断演化和发展。其中,Go语言作为一种由Google开发的开源编程语言,自问世以来备受关注并广泛使用。与其他编程语言相比,Go
Go语言与其他编程语言的比较分析
2024-03-13

分析Go语言的特点与编程语言的本质

标题:深度分析Go语言的特点与编程语言的本质编程语言是人类用于编写计算机程序的一种符号系统,不同的编程语言具有不同的特点和设计哲学。其中,Go语言作为一种新兴的编程语言,具有独特的特点,促使人们重新思考编程语言的本质。在本文中,将深入分析
分析Go语言的特点与编程语言的本质
2024-03-07

汇编语言的函数式编程实例分析

这篇文章主要介绍了汇编语言的函数式编程实例分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇汇编语言的函数式编程实例分析文章都会有所收获,下面我们一起来看看吧。一切都是为了消除副作用要了解函数式编程,我们需要首
2023-06-27

C++语言的各种编程技巧分析

这篇文章主要介绍“C++语言的各种编程技巧分析”,在日常操作中,相信很多人在C++语言的各种编程技巧分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C++语言的各种编程技巧分析”的疑惑有所帮助!接下来,请跟
2023-06-17

C语言中程序编译系统的示例分析

这篇文章主要介绍C语言中程序编译系统的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!程序的翻译环境和执行环境在ANSI C的任何一种实现中,存在两个不同的环境 :第1种是翻译环境,在这个环境中源代码被转换为可
2023-06-29

比较Go语言和Java的差异:语法和编程模式的分析

Go语言与Java的差异:语法与编程模型Go语言和Java都是现代流行的编程语言,具有许多相似之处,但也有许多差异。这些差异主要体现在语法和编程模型上。语法1. 变量声明Go语言中,变量声明需要使用var关键字,而Java中则不需要
比较Go语言和Java的差异:语法和编程模式的分析
2024-02-01

Go语言编程优势分析与应用指南

Go语言作为一种相对年轻的编程语言,却在近年来迅速崛起,备受程序员们的关注和青睐。它以其简洁的语法、高效的并发机制和出色的性能而闻名于业内。本文将分析Go语言的编程优势,并结合具体的代码示例进行应用指南,帮助读者更好地理解和运用这门语言。
Go语言编程优势分析与应用指南
2024-03-11

使用Python编程语言进行实践的示例分析

这篇文章将为大家详细讲解有关使用Python编程语言进行实践的示例分析,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。其实Python真的挺有意思的,用Python用做过不少有趣好玩的事儿,这
2023-06-02

如何进行Go编程语言的简单分析

今天就跟大家聊聊有关如何进行Go编程语言的简单分析,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Go 是一门用于并发编程的命令式编程语言,它主要由创造者 Google 进行开发,最初
2023-06-16

C语言编程中if的用法是什么

C语言编程中if的用法是什么,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。顺序,选择和循环是程序的三大基本结构,无论多么复杂的程序,都是用这三种结构写出来的。i
2023-06-03

Go语言中interface语法与使用实例分析

这篇“Go语言中interface语法与使用实例分析”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Go语言中interfac
2023-07-02

go语言中的协程实例分析

这篇文章主要讲解了“go语言中的协程实例分析”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“go语言中的协程实例分析”吧!协程的特点1.该任务的业务代码主动要求切换,即主动让出执行权限2.发生
2023-07-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录