https://segmentfault.com/a/1190000012646488  https://yangbo5207.github.io/wutongluo/

说明:此处只是记录阅读前端基础进阶的理解和总结,如有需要请阅读上面的链接

一、this的指向在执行上下文的创建阶段确定,即是在函数的调用阶段确定的,因此不同的调用方式this的指向可能不同

    var a = 10;
var obj = {
a: 20,
GetA:function(){
console.log(this.a);
}
} obj.GetA(); //
var f = obj.GetA;
f();//
f()独立调用,this指向全局环境;obj.GetA()不是独立调用,this指向函数所属的对象。

关于this指向的结论:在一个函数的全局上下文中,this用调用者提供,由函数的调用方式决定。如果函数被调用时被某一个对象所拥有(即写法如obj,GetA()),那么函数内部的this指向改对象;
如果函数是独立调用,那么内部的this指向undifined。在非严格模式下当this指向undefined时,它会被自动指向全局对象。 注意事项:全局环境中的this指向全局环境本身,this指向在函数执行过程中不能改变,所以给this赋值会报错,如this=obj; 说明:this指代的是对象,函数中的this并不是表示函数作用域。如下例子刚开始以为输出a=10,a表示函数fn的作用范围,其实不是a表示的是全局对象。
    var a = 20;
function fn() {
var a=10;
function foo() {
console.log(this.a);
}
foo();//独立调用,所以输出结果还是20
}
fn();

严格模式下的foo()是独立调用,this指向undifined,故this.a报错,而window.foo()不是独立调用,this指向window,所以this.a=20,故结果40

 'use strict';
var a = 20;
function foo() {
var a = 1;
var obj = {
a: 10,
c: this.a + 20,
fn: function () {
return this.a;
}
}
return obj.c; }
console.log(window.foo()); //
console.log(foo()); // TypeError: this is undefined
三、使用call,apply显示指定this

JavaScript可使用call和apply方法手动设置函数中this的指向,每个函数都支持call和apply方法

    var a = 10;
var obj={
a:10
}; function fn() {
console.log(this.a);
} fn.call(obj);//

call和apply的用法

第一个参数为this的指向,剩下的参数是函数本身的参数,区别是call方法中函数参数是一个一个传入,apply方法以数组方式传入

   var a = 10;
var obj={
a:10
}; function fn(num1,num2) {
console.log(num1+num2+this.a);
} fn.call(obj, 1, 2); //
fn.apply(obj, [1, 2]); //

四 call/apply的用途

1.将类数组转换为数组

 function add(num1, num2, mum3) {
console.log(arguments); //打印出类数组原来的样子Arguments { 0: 1, 1: 2, 2: 3, … }
var arg = [].slice.call(arguments); //把类数组转换成数组 Array [ 1, 2, 3 ]
console.log(arg);
}
add(1, 2, 3);

2.根据自己需要灵活指定this

var foo = {
name: 'joker',
showName: function () {
console.log(this.name);
}
}
var bar = {
name: 'rose'
}
foo.showName.call(bar);//rose

3.实现继承

    //定义父级构造函数
var Person = function (name, age) {
this.name = name;
this.age = age;
} //定义子类构造函数
var Student = function (name, age, high) {
Person.call(this, name, age);
this.high = high;
} Student.prototype.message = function () {
console.log('name:' + this.name + 'age:' + this.age + 'high:' + this.high);
} var stu = new Student('xiaoming', 10, '150cm');
stu.message(); //name:xiaomingage:10high:150cm

要理解上面的例子,首先要搞清楚构造函数中this的指向。this指向是在函数调用时确定的,所以要弄明白构造函数中的this指向就要搞清楚new之后经历了什么。

调用new运算符之后会经历一下阶段:

1)创建一个新的对象

2)将构造函数中的this指向这个新的对象

3)执行构造函数的代码为这个新对象添加属性,方法等

4)返回新对象

上面的例子在Student的构造函数中利用call方法把新创建的对象传递给Person,从而给新对象添加name,age属性,也就相当于实现了继承。

4.保存this指向正确的对象

    var a = 20;
var obj = {
a:10,
getA:function(){
setTimeout(function(){
console.log(this.a);
},1000);
}
}
obj.getA();//

上例本意是想打印出对象obj中a的值,但是由于setTimeout函数的存在,导致实际执行的时候this指向了全局对象,所以打印结果是20。

注意:setTimeout的作用是向JavaScript注册函数,并在指定时间之后执行注册的函数,因此等到函数运行时这个函数就相当于是独立调用,this在费严格模式下自动指向全局对象。

最简单的改法是利用闭包把this指向的对象保存下来

 var a = 20;
var obj = {
a: 10,
getA: function () {
var self = this;
setTimeout(function () {
console.log(self.a);
}, 1000);
}
}
obj.getA();//

另一个方式是使用call或者apply封装一个bind方法,确保函数指向obj

 var a = 20;
var obj = {
a: 10,
getA: function () { setTimeout(bind(function () {
console.log(this.a);
},this), 1000);
}
} function bind(fn, obj) {
return function () {
fn.call(obj); //或者fn.apply(obj);
}
} obj.getA();//

还可以用JavaScript自带的bind方法,作用是把函数fn.bind(obj)的this指向绑定到obj

    var a = 20;
var obj = {
a: 10,
getA: function () { setTimeout(function () {
console.log(this.a);
}.bind(this), 1000);
}
} obj.getA();//
 

前端基础进阶(五):全方位解读this的更多相关文章

  1. 前端基础进阶(十五):详解 ES6 Modules

    对于新人朋友来说,想要自己去搞定一个ES6开发环境并不是一件容易的事情,因为构建工具的学习本身又是一个非常大的方向,我们需要花费不少的时间才能掌握它. 好在慢慢的开始有大神提供了一些非常简单易懂,学习 ...

  2. 前端基础进阶(七)-前端工程师最容易出错的问题-this关键字

    我们在学习JavaScript的时候,因为对一些概念不是很清楚,但是又会通过一些简洁的方式把它给记下来,那么这样自己记下来的概念和真正的概念产生了很强的偏差. 当然,还有一些以为这个是对的,还会把它发 ...

  3. 前端基础进阶(十四):es6常用基础合集

    在实际开发中,ES6已经非常普及了.掌握ES6的知识变成了一种必须.尽管我们在使用时仍然需要经过babel编译. ES6彻底改变了前端的编码风格,可以说对于前端的影响非常巨大.值得高兴的是,如果你熟悉 ...

  4. 前端基础进阶(十一):详细图解jQuery对象,以及如何扩展jQuery插件

    早几年学习前端,大家都非常热衷于研究jQuery源码.我还记得当初从jQuery源码中学到一星半点应用技巧的时候常会有一种发自内心的惊叹,“原来JavaScript居然可以这样用!” 虽然随着前端的发 ...

  5. 前端基础进阶(六):在chrome开发者工具中观察函数调用栈、作用域链与闭包

    在前端开发中,有一个非常重要的技能,叫做断点调试. 在chrome的开发者工具中,通过断点调试,我们能够非常方便的一步一步的观察JavaScript的执行过程,直观感知函数调用栈,作用域链,变量对象, ...

  6. 前端基础进阶之Promise

    前言 Promise的重要性我认为我没有必要多讲,概括起来说就是必须得掌握,而且还要掌握透彻.这篇文章的开头,主要跟大家分析一下,为什么会有Promise出现. 在实际的使用当中,有非常多的应用场景我 ...

  7. 转【前端基础进阶之Promise】

    前言 Promise的重要性我认为我没有必要多讲,概括起来说就是必须得掌握,而且还要掌握透彻.这篇文章的开头,主要跟大家分析一下,为什么会有Promise出现. 在实际的使用当中,有非常多的应用场景我 ...

  8. 前端基础进阶(十三):透彻掌握Promise的使用,读这篇就够了

    Promise的重要性我认为我没有必要多讲,概括起来说就是必须得掌握,而且还要掌握透彻.这篇文章的开头,主要跟大家分析一下,为什么会有Promise出现. 在实际的使用当中,有非常多的应用场景我们不能 ...

  9. 前端基础进阶(六)-大厂面试题问题:循环闭包与setTimeout

    我在上一篇闭包的文章中留下了一个关于setTimeout与循环闭包的思考题. 利用闭包,修改下面的代码,让循环输出的结果依次为1, 2, 3, 4, 5 for (var i = 1; i <= ...

随机推荐

  1. ado.net EF学习系列----深入理解查询延迟加载技术(转载)

    ado.net EF是微软的一个ORM框架,使用过EF的同学都知道EF有一个延迟加载的技术. 如果你是一个老鸟,你可能了解一些,如果下面的学习过程中哪些方面讲解的不对,欢迎批评指教.如果一个菜鸟,那我 ...

  2. [No0000B0]ReSharper操作指南1/16-入门与简介

    安装指南 在安装之前,您可能需要检查系统要求. ReSharper是一个VisualStudio扩展.它支持VisualStudio2010,2012,2013,2015和2017.安装完成后,您将在 ...

  3. [No0000C9]神秘的掐指一算是什么?教教你也会

    很多朋友看到传说中诸葛亮以及那些聪明人掐指一算,惊叹不已.那些人以“察天地之理.通鬼神之志”,每次占卜时,做一大堆的神秘仪式,然后掐指一算,便大有“乾坤尽收在手”的感觉.在普通人眼里,他们的手神秘异常 ...

  4. 【视频】dx dy的意思 微分的定义 导数符号的意思

    视频解说 http://www.bilibili.com/video/av8565224/

  5. 用em包图片记

    像这种很常见的字体旁边带图标的,很多都是用em的background去实现. <li><em class="wx"></em><a hre ...

  6. Will vs Be Going To vs Present Continuous: Talk About the Future in English

    https://www.youtube.com/watch?v=UISiuiPd_FY will 说话的当下决定的将来要做什么,in the moment be going to 有意图去做,但没有计 ...

  7. day3_字典

    一.说明 字典的每个键值(key=>value)对用冒号(:)分割,每个对之间用逗号(,)分割,整个字典包括在花括号({})中 ,格式如下所示: dict = {key1:value1,key2 ...

  8. Struts2漏洞检查工具

  9. mongo 的简单查询语法

    小白的我对MONGO的一些语句搜集用于区别mysql及一些小常识 pymongo 语法按照id进行倒序操作db.news.find().limit(20).sort([("_id" ...

  10. 洛谷P4426 毒瘤 [HNOI/AHOI2018] 虚树+树上dp

    正解:虚树+树上dp 解题报告: 传送门! 首先解释一下题意趴,,,语文70pts选手已经开始看不懂题辣QAQ 大概就是个给一个图,求独立集方案,且保证图是联通的,边的数量最多只比点多10 首先思考如 ...