专栏首页大闲人柴毛毛稳扎稳打JS——this

稳扎稳打JS——this

this的值是在运行时确定的

JS中的this究竟代表什么,这是在程序运行时根据上下文环境确定,可以分为以下几种情况。

1. 全局作用域中的this

在全局作用域中,this指向window对象。

console.log(this);//指向window对象

this.x = 5//在全局作用域内创建一个x
//与this.x = 5的等价情况:
//var x = 5;
//x = 5;
  • 在全局作用域中执行var x=5,其实是为window对象创建一个属性x,并令其等于5。
  • 若定义变量时不加var,JS会认为该变量为全局变量,会将其当作window对象的属性。

2. 函数中的this

JS中函数有两种,直接调用的函数称为普通函数,通过new创建对象的函数称为构造函数。

2.1 构造函数中的this

构造函数的this指向它所创建的对象,如:

function Person(name){
    this.name = name;//this指向该函数创建的对象person
}
var person = new Person("chaimm");

2.2 普通函数中的this

普通函数的this指向window对象。 若上述例子,直接执行Perosn函数,则其中this代表window对象,因此该函数执行后会创建一个全局的name。

function Person(name){
    this.name = name;//this指向window
}
Person("chai");//当作普通函数执行,this指向window对象

3. 对象中的this

对象中的this指向当前对象,如:

var person = {
    name : "chaimm",
    getName : function(){
        return this.name;
    }
}

上述代码中this指向函数getName所属的对象。 但是,如果一个对象的函数中又嵌套了一个函数,这个函数的this指向的却是window,而并不是其外层的对象。

var person = {
    name : "chaimm",
    setName : function(name){
        (function(name){
            this.name = name; //此时this并不代表person对象,而是代表window对象
        })(name);
    }
}

上述示例中,person对象中有一个getName函数,而getName函数内部又有一个函数,这个函数内部的this指向window对象,而非person对象,这是JS的一个bug!一般作如下处理,规避这个bug:

var person = {
    name : "chaimm",
    setName : function(name){
        var thar = this;//将this赋给that
        (function(name){
            that.name = name;//此时that指向person对象
        })(name);
    }
}

我们在person对象的第一层函数中,将this赋给局部变量that,然后在第二层函数中使用that,此时that指向person对象,可对person的属性进行操作。

  • 注意:若将一个对象中的函数赋给一个变量后,再通过该变量调用这个函数,此时该函数中的this指向window,而非该对象,如下所示:
var person = {
    name : "chaimm",
    getName : function(){
        return this.name;
    }
}

//将getName函数赋给一个新的变量
var newGetName = person.getName;
//通过新的变量调用这个函数,这个函数中的this将指向window
newGetName();//若全局作用域中没有name,则将返回undefined

4. 用call和apply函数给this开挂

这两个函数都能手动指定被调用函数内部的this指向哪个对象。

//定义一个构造函数
var Person = function(name){
    this.name = "";
    this.setName = function(name){
        this.name = name;
    }
}

//创建两个对象
var personA = new Person("A");
var personB = new Person("B");

//使用personA的setName函数去修改personB的name属性
personA.setName.apply(personB,["C"]);
  • apply用法 对象A.函数名.apply(对象B, 参数列表); 当对象B作为apply的第一个参数传给apply时,对象A的函数中this就指向了对象B,此时对象A的该函数对this的操作将会作用在对象B上,由此实现了用对象A去调用对象B的函数。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 稳扎稳打JavaScript(二)——图解对象内存模型

    1. 什么是JS对象? 在JS中,对象是一组无序属性的集合。其中,属性可以是基本数据类型、引用类型、函数。如下面这个对象的例子: var chai={ ...

    大闲人柴毛毛
  • 三分钟理解“享元模式”——设计模式轻松掌握

    享原模式的官方定义: 运用共享技术有效地支持大量细粒度的对象。 大白话讲: 一个类它可能生成好多对象,但这些对象根据属性值的不同一共分成N类,每种类型中属性值都...

    大闲人柴毛毛
  • 稳扎稳打JS——“对象”

    一切皆“对象” JS中一切皆“对象” “对象”是属性的集合,而属性又是对象。既然属性又是对象,那么一个对象的属性也可以拥有属性,如: //定义一个函数getN...

    大闲人柴毛毛
  • 你要的对象都在这啦|how to create Object in JavaScript?

    小伙伴们,大家好,今天我们来说一下Javascript中创建对象的几种方式,请看文章。

    石璞东
  • 稳扎稳打JavaScript(二)——图解对象内存模型

    1. 什么是JS对象? 在JS中,对象是一组无序属性的集合。其中,属性可以是基本数据类型、引用类型、函数。如下面这个对象的例子: var chai={ ...

    大闲人柴毛毛
  • 来看三个问题

    是否允许在HTTP请求时,返回原始请求体数据字节,默认为false(GET or HEAD or 上传文件请求除外)。

    我的小碗汤
  • 恭喜你获得治疗this“皮”的详细药方

    众所周知,this在JavaScript中的指向一直很难让人理解,想要学好JavaScript,this也是我们必须要搞清楚的。其实,this并没有那么难,本文...

    童欧巴
  • Flutter基础widgets教程-CupertinoNavigationBar篇

    青年码农
  • 十一、详解面向对象

    如果要我总结一下学习前端以来我遇到了哪些瓶颈,那么面向对象一定是第一个会想到的。尽管现在对于面向对象有了一些的了解,但是当初那种似懂非懂的痛苦,依然历历在目。

    用户6901603
  • 如何用 vue 制作一个探探滑动组件

    前言 嗨,说起探探想必各位程序汪都不陌生(毕竟妹子很多),能在上面丝滑的翻牌子,探探的的堆叠滑动组件起到了关键的作用,下面就来看看如何用vue写一个探探的堆叠组...

    程序员宝库

扫码关注云+社区

领取腾讯云代金券