对于前端程序媛(员)来说,this这个机制应用的地方是很多的,所以搞懂是必要的,不熟练使用this将遇到一些困惑,下面是一些关于this的学习心得分享,希望大家可以一起学习:

1,this并不是指向自身

首先来看一段代码

     function foo(num) {
console.log("foo:" + num); //0 1 2 3 4
console.log(this); //window对象
this.count++; //记录foo被调用的次数
}
foo.count = 0; //count 属于foo的属性 并不属于window
var i;
for (i = 0; i < 10; i++) {
if (i < 5) {
foo(i);
}
}
console.log(foo.count) //

很多人认为 :函数内部的this指向当前这个函数,这是错误的

对于这个例子来说,一种解决方法是使用foo标识符代替this来指向自身

  function foo(num) {
console.log("foo:" + num); //0 1 2 3 4
console.log(this); //window对象
foo.count++; //记录foo被调用的次数
}
foo.count = 0; //count 属于foo的属性 并不属于window
var i;
for (i = 0; i < 10; i++) {
if (i < 5) {
foo(i);
}
}
console.log(foo.count) //

另一种方法是没有规避了this使用

function foo(num) {
console.log("foo:" + num); //0 1 2 3 4
console.log(this); //window对象
this.count++; //记录foo被调用的次数
}
foo.count = 0; //count 属于foo的属性 并不属于window
var i;
for (i = 0; i < 10; i++) {
if (i < 5) {
foo.call(foo, i) //使用call() 确保this 指向函数foo
}
}
console.log(foo.count) //

2,this指向函数作用域(并不是一直都是对的)

需要知道的一点是this在任何情况都不会指向函数的词法作用域,因为作用域是无法通过js代码访问到的,它存在于引擎内部

下面这个代码很好的说明了这个问题

    function foo() {
var a = 2;
this.bar(); // this是window
} function bar() {
console.log(this); //window
console.log(this.a); //undefined
}
foo();

this究竟是指向什么呢?this是在运行时进行绑定的,并不是在编写时绑定的,它的上下文取决于函数调用时的各种条件,this绑定和函数声明的位置没有任何关系,只取决于函数的调用方式。

当函数调用时,会创建一个执行上下文,这个上下文会记录函数在哪里调用了,函数的调用方法,函数参数等信息,this就是该上下文的一个属性,会在函数执行的过程中用到。

3,函数在执行过程中this是如何绑定的

寻找函数的调用位置,其实有的很明显,而某些编程模式会因此函数的真正的调用位置

分析调用栈

  function baz() {
//当前的调用栈是baz
//因此,当前的调用位置是全局作用域
console.log("baz");
bar(); //bar的调用位置
} function bar() {
//当前的调用栈是baz->bar
//因此,当前的调用位置是baz
console.log("bar");
foo(); //foo的调用位置
} function foo() {
//当前的调用栈是baz->bar->foo
//因此,当前的调用位置是bar
console.log("foo");
}
baz(); //baz的调用位置

分析调用栈是,决定了this的绑定

 4,this的绑定规则

①默认绑定:独立函数调用  ,在下面的代码中,foo()是直接使用不带任何修饰符的函数引用进行调用的,因此只能使用默认绑定

  function foo() {
console.log(this.a); //this指向window
}
var a = 2;
foo(); //

但是在严格模式下,全局对象将无法使用默认绑定,因此this会绑定到undefined

     function foo() {
'use strict'
console.log(this); //undefined
console.log(this.a); //报错
}
var a = 2;
foo();

但是有个微妙的非常重要的情况,虽然this的邦定规则完全取决于调用位置,但是只有foo运行在非严格模式下是默认绑定才能绑定到全局对象,但在严格模式下,于foo的调用位置无关了

   function foo() {
console.log(this); //window
console.log(this.a); //
}
var a = 2; (function() {
'use strict'
console.log(this);//undefined
foo();
})()

②隐式绑定

 function foo() {
console.log(this); //obj
console.log(this.a); //
}
var obj = {
a: 2,
foo: foo
}
obj.foo();

当foo被调用的时候,它的落脚点确实指向obj对象,当函数引用有上下文对象时,隐式绑定规则会把函数调用中的this邦定到这个上下文对象中,因此,调用foo()时,this被绑定到obj,因此,this.a和obj.a是一样的

③显示绑定

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

.call(...)和apply(...)方法实现显示绑定

他们的第一个参数是一个对象,不是对象的会利用自己的包装对象转换成对象(new Number()  ,new String() ,new Boolen()等),他们会把这个对象绑定到this,接着在调用函数时指定这个this,因为你可以直接指定this的绑定对象,因此我们称为显式绑定

④new绑定

     function foo(a) {
this.a = a;
}
var bar = new foo(2);
console.log(bar.a);//

构造函数是使用new操作时会被调用的函数。他们并不属于类,也不会实例化一个类,实际上,甚至不能说他们是一直特殊的函数类型,他们只是用new操作费调用的普通函数而已。

使用new来调用函数,会自动执行下面操作:

1,创建(或者构造一个)全新的对象

2,这个对象会被执行【原型】连接

3,这个新对象会被绑定到函数调用的this,

4,如果函数没有返回其他对象,那么new表达式中的函数调用会自动返回这个新对象

上面的代码,使用new调用foo()时,我们会构造一个新的对象并把它绑定到foo()调用中的this上,new是最后一个可以影响函数调用时this邦定行为的方法,我们称之为new绑定

this 相关的更多相关文章

  1. 嵌入式单片机STM32应用技术(课本)

    目录SAIU R20 1 6 第1页第1 章. 初识STM32..................................................................... ...

  2. java中的字符串相关知识整理

    字符串为什么这么重要 写了多年java的开发应该对String不陌生,但是我却越发觉得它陌生.每学一门编程语言就会与字符串这个关键词打不少交道.看来它真的很重要. 字符串就是一系列的字符组合的串,如果 ...

  3. SQL Server相关书籍

    SQL Server相关书籍 (排名不分先后) Microsoft SQL Server 企业级平台管理实践 SQL Server 2008数据库技术内幕 SQL Server性能调优实战 SQL S ...

  4. dotNET跨平台相关文档整理

    一直在从事C#开发的相关技术工作,从C# 1.0一路用到现在的C# 6.0, 通常情况下被局限于Windows平台,Mono项目把我们C#程序带到了Windows之外的平台,在工作之余花了很多时间在M ...

  5. 在ASP.NET Core应用中如何设置和获取与执行环境相关的信息?

    HostingEnvironment是承载应用当前执行环境的描述,它是对所有实现了IHostingEnvironment接口的所有类型以及对应对象的统称.如下面的代码片段所示,一个HostingEnv ...

  6. virtualbox linux虚拟机相关

    linux虚拟机设置为静态IP 在virtualbox中安装好linux虚拟机后,如果采用的是NAT方式的话,linux虚拟机默认采用dhcp方式自动上网,而且用的是NetworkManager服务而 ...

  7. WebGIS中等值面展示的相关方案简析

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 等值面是气象.环保等相关项目上常用到的效果展示.在传统的CS项 ...

  8. .NET同步与异步之相关背景知识(六)

    在之前的五篇随笔中,已经介绍了.NET 类库中实现并行的常见方式及其基本用法,当然.这些基本用法远远不能覆盖所有,也只能作为一个引子出现在这里.以下是前五篇随笔的目录: .NET 同步与异步之封装成T ...

  9. zookeeper集群的搭建以及hadoop ha的相关配置

    1.环境 centos7 hadoop2.6.5 zookeeper3.4.9 jdk1.8 master作为active主机,data1作为standby备用机,三台机器均作为数据节点,yarn资源 ...

  10. mysql 5.7中的用户权限分配相关解读!

    这篇文章主要介绍了MySQL中基本的用户和权限管理方法,包括各个权限所能操作的事务以及操作权限的一些常用命令语句,是MySQL入门学习中的基础知识,需要的朋友可以参考下 一.简介 各大帖子及文章都会讲 ...

随机推荐

  1. 如何使用PowerShell批量删除Office 365的用户

    概述 本文将演示如何在必要的时候(例如在测试环境),通过PowerShell脚本批量删除Office 365的用户,首先需要通过Get-MsolUser的命令(并且配合筛选条件)获取到符合条件的用户列 ...

  2. LeetCode(119):杨辉三角 II

    Easy! 题目描述: 给定一个非负索引 k,其中 k ≤ 33,返回杨辉三角的第 k 行. 在杨辉三角中,每个数是它左上方和右上方的数的和. 示例: 输入: 3 输出: [1,3,3,1] 进阶: ...

  3. LeetCode(110):平衡二叉树

    Easy! 题目描述: 给定一个二叉树,判断它是否是高度平衡的二叉树. 本题中,一棵高度平衡二叉树定义为: 一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过1. 示例 1: 给定二叉树 [3, ...

  4. 广工赛-hdu6468构造十叉树

    是个以前没见过的模板题.. 我用比较复杂度方式过掉了.. 构造一个十叉树(有点trie的味道)来存数字,然后字典序就是先序遍历的结果 #include<bits/stdc++.h> usi ...

  5. python 内置数据类型之数字

    目录: 1.2. 数字 1.2.1. 数字类型 1.2.2. 浮点数 1.2.3. 进制记数 1.2.4. 设置小数精度 1.2.5. 分数 1.2.6. 除法 1.2 数字   1.2.1 数字类型 ...

  6. spring cloud 使用ribbon简单处理客户端负载均衡

    假如我们的multiple服务的访问量剧增,用一个服务已经无法承载, 我们可以把Hello World服务做成一个集群. 很简单,我们只需要复制Hello world服务,同时将原来的端口8762修改 ...

  7. IDEA复制项目操作

  8. ElasticSearch(五) Elasticsearch-jdbc实现MySQL同步到ElasticSearch

    https://www.cnblogs.com/wt645631686/p/8274722.html

  9. mysql恢复ibd文件

    1.将原表删除,包括ibd和frm文件 2.重新创建表结构. 3.丢弃表空间 alter table tableName discard tablespace; 4.将要恢复的ibd文件拷贝到数据库目 ...

  10. lojround3

    A.绯色 IOI(开端) 首先注意到是完全图,数据范围又很大,肯定要观察一些性质 我们化简一下式子 发现其实是要求simga(xixj)最大 那么结论就很好想了 最大的和次大的第三大的连一起...然后 ...