图解javascript this指向什么?

本文详细解析了JavaScript中this关键字的工作原理及其在不同场景下的表现,包括作为对象方法、普通函数、构造函数调用以及使用apply或call时this的绑定情况。

出处:http://www.cnblogs.com/isaboy/
声明:本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

JavaScript 是一种脚本语言,支持函数式编程、闭包、基于原型的继承等高级功能。JavaScript一开始看起来感觉会很容易入门,但是随着使用的深入,你会发现JavaScript其实很难掌握,有些基本概念让人匪夷所思。其中JavaScript 中的 this 关键字,就是一个比较容易混乱的概念,在不同的场景下,this会化身不同的对象。有一种观点认为,只有正确掌握了 JavaScript 中的 this 关键字,才算是迈入了 JavaScript 这门语言的门槛。在主流的面向对象的语言中(例如Java,C#等),this 含义是明确且具体的,即指向当前对象。一般在编译期绑定。而 JavaScript 中this 在运行期进行绑定的,这是JavaScript 中this 关键字具备多重含义的本质原因。

  JavaScript由于其在运行期进行绑定的特性,JavaScript 中的 this 可以是全局对象、当前对象或者任意对象,这完全取决于函数的调用方式。JavaScript 中函数的调用有以下几种方式:作为对象方法调用,作为函数调用,作为构造函数调用,和使用 apply 或 call 调用。常言道,字不如表,表不如图。为了让人更好的理解JavaScript this 到底指向什么?下面用一张图来进行解释:

this指向

  上图我称之为”JavaScript this决策树”(非严格模式下)。下面通过例子来说明这个图如何来帮助我们对this进行判断:

1 var point = { 
 2  x : 0, 
 3  y : 0, 
 4  moveTo : function(x, y) { 
 5      this.x = this.x + x; 
 6      this.y = this.y + y; 
 7      } 
 8  };

 9 //决策树解释:point.moveTo(1,1)函数不是new进行调用,进入否决策,
10 //是用dot(.)进行调用,则指向.moveTo之前的调用对象,即point
11 point.moveTo(1,1); //this 绑定到当前对象,即point对象
复制代码

  point.moveTo()函数在 “JavaScript this决策树”中进行判定的过程是这样的:

  1)point.moveTo函数调用是用new进行调用的么?这个明显不是,进入“否”分支,即函数是否用dot(.)进行调用?;

  2)point.moveTo函数是用dot(.)进行调用的,即进入“是”分支,即这里的this指向point.moveTo中.之前的对象point;

  图解point.moveTo函数的this指向什么的解析图如下图所示:
dot调用this指向

  再举例,看下面的代码:

1 function func(x) { 
2  this.x = x; 
3  } 
4 func(5); //this是全局对象window,x为全局变量
5 //决策树解析:func()函数是用new进行调用的么?为否,进入func()函数是用dot进行调用的么?为否,则 this指向全局对象window
6 x;//x => 5

  func()函数在 “JavaScript this决策树”中进行判定的过程是这样的:

  1)func(5)函数调用是用new进行调用的么?这个明显不是,进入“否”分支,即函数是否用dot(.)进行调用?;

  2)func(5)函数不是用dot(.)进行调用的,即进入“否”分支,即这里的this指向全局变量window,那么this.x实际上就是window.x;

  图解func函数的this指向什么的解析图如下图所示:
  不是new和dot的方法调用this指向
  针对作为函数直接调用的方式,下面看一个复杂的例子:

1 var point = { 
 2  x : 0, 
 3  y : 0, 
 4  moveTo : function(x, y) { 
 5      // 内部函数
 6      var moveX = function(x) { 
 7      this.x = x;//this 指向什么?window
 8     }; 
 9     // 内部函数
10     var moveY = function(y) { 
11     this.y = y;//this 指向什么?window
12     }; 
13     moveX(x); 
14     moveY(y); 
15     } 
16  }; 
17  point.moveTo(1,1); 
18  point.x; //=>0 
19  point.y; //=>0 
20  x; //=>1 
21  y; //=>1

  point.moveTo(1,1)函数实际内部调用的是moveX()和moveY()函数, moveX()函数内部的this在 “JavaScript this决策树”中进行判定的过程是这样的:

  1)moveX(1)函数调用是用new进行调用的么?这个明显不是,进入“否”分支,即函数是否用dot(.)进行调用?;

  2)moveX(1)函数不是用dot(.)进行调用的,即进入“否”分支,即这里的this指向全局变量window,那么this.x实际上就是window.x;

  下面看一下作为构造函数调用的例子:

1 function Point(x,y){ 
2     this.x = x; // this ?
3     this.y = y; // this ?
4  }
5 var np=new Point(1,1);
6 np.x;//1
7 var p=Point(2,2);
8 p.x;//error, p是一个空对象undefined
9 window.x;//2

  Point(1,1)函数在var np=new Point(1,1)中的this在 “JavaScript this决策树”中进行判定的过程是这样的:

  1)var np=new Point(1,1)调用是用new进行调用的么?这个明显是,进入“是”分支,即this指向np;

  2)那么this.x=1,即np.x=1;

  Point(2,2)函数在var p= Point(2,2)中的this在 “JavaScript this决策树”中进行判定的过程是这样的:

  1)var p= Point(2,2)调用是用new进行调用的么?这个明显不是,进入“否”分支,即函数是否用dot(.)进行调用?;

  2)Point(2,2)函数不是用dot(.)进行调用的?判定为否,即进入“否”分支,即这里的this指向全局变量window,那么this.x实际上就是window.x;

  3)this.x=2即window.x=2.

  最后看一下函数用call 和apply进行调用的例子:

1 function Point(x, y){ 
 2     this.x = x; 
 3     this.y = y; 
 4     this.moveTo = function(x, y){ 
 5         this.x = x; 
 6         this.y = y; 
 7     } 
 8  } 
 9 
10 var p1 = new Point(0, 0); 
11 var p2 = {x: 0, y: 0}; 
12 p1.moveTo.apply(p2, [10, 10]);//apply实际上为p2.moveTo(10,10)
13 p2.x//10

  p1.moveTo.apply(p2,[10,10])函数在 “JavaScript this决策树”中进行判定的过程是这样的:

  我们知道,apply 和 call 这两个方法异常强大,他们允许切换函数执行的上下文环境(context),即 this 绑定的对象。p1.moveTo.apply(p2,[10,10])实际上是p2.moveTo(10,10)。那么p2.moveTo(10,10)可解释为:

  1)p2.moveTo(10,10)函数调用是用new进行调用的么?这个明显不是,进入“否”分支,即函数是否用dot(.)进行调用?;

  2)p2.moveTo(10,10)函数是用dot(.)进行调用的,即进入“是”分支,即这里的this指向p2.moveTo(10,10)中.之前的对象p2,所以p2.x=10;

  关于JavaScript函数执行环境的过程,IBM developerworks文档库中的一段描述感觉很不错,摘抄如下:

“JavaScript 中的函数既可以被当作普通函数执行,也可以作为对象的方法执行,这是导致 this 含义如此丰富的主要原因。一个函数被执行时,会创建一个执行环境(ExecutionContext),函数的所有的行为均发生在此执行环境中,构建该执行环境时,JavaScript 首先会创建 arguments变量,其中包含调用函数时传入的参数。接下来创建作用域链。然后初始化变量,首先初始化函数的形参表,值为 arguments变量中对应的值,如果 arguments变量中没有对应值,则该形参初始化为 undefined。如果该函数中含有内部函数,则初始化这些内部函数。如果没有,继续初始化该函数内定义的局部变量,需要注意的是此时这些变量初始化为 undefined,其赋值操作在执行环境(ExecutionContext)创建成功后,函数执行时才会执行,这点对于我们理解 JavaScript 中的变量作用域非常重要,鉴于篇幅,我们先不在这里讨论这个话题。最后为 this变量赋值,如前所述,会根据函数调用方式的不同,赋给 this全局对象,当前对象等。至此函数的执行环境(ExecutionContext)创建成功,函数开始逐行执行,所需变量均从之前构建好的执行环境(ExecutionContext)中读取。”

  理解这段话对于理解Javascript函数将大有好处。

内容概要:本文介绍了ER腐蚀速率探头在多个工业领域中的应用及其重要性。ER探头具有强大的环境适应能力,可在高温、高压、低温(如-162℃)及高盐度等恶劣条件下稳定运行,广泛应用于石油天然气、化工炼化、电力等行业。在油气管道中,尤其适用于含硫化氢的酸性环境和海上平台导管架的潮差区,可实时监测腐蚀速率,频率高达每30秒一次,当腐蚀速率超过0.01mm/年时及时预警,便于调整缓蚀剂或阴极保护参数。在化工装置中,探头用于催化裂化反应器、换热器、再沸器和酸碱储罐等关键设备,实现不停工在线监测,降低检修成本并防止泄漏事故。电力行业中,ER探头用于锅炉水冷壁和凝汽器管束的腐蚀监控,结合水质分析优化水处理方案,延长设备寿命。; 适合人群:从事石油天然气、化工、电力等行业设备安全监测、腐蚀防护及相关技术研发的工程技术人员;从事管道运维、材料防护或安全管理的专业人员;; 使用场景及目标:① 实时在线监测关键设备和管道的腐蚀速率,预防穿孔泄漏事故;② 结合工艺参数优化缓蚀剂、杀菌剂使用及水处理方案;③ 在高温、高压、低温等复杂工况下实现稳定可靠的腐蚀评估; 阅读建议:本资料侧重于ER探头的实际应用场景和技术价值,使用者应结合具体工况理解其部署位置与数据解读方法,并可参考河南科捷制造有限公司的技术支持进一步落地实施。
摘 要 当今社会进入了科技进步、经济社会快速发展的新时代。国际信息和学术交流也不断加强,计算机技术对经济社会发展和人民生活改善的影响也日益突出,人类的生存和思考方式也产生了变化。传统购物方式采取了人工的管理方法,但这种管理方法存在着许多弊端,比如效率低下、安全性低以及信息传输的不准确等,同时由于购物管理信息系统中会形成众多的个人文档和信息系统数据,通过人工方法对购物信息进行集中管理会形成检索、更改和维护等较为麻烦的管理问题,同时由于当下人民群众对购物管理的需求也日益高涨,各级单位需要继续开展全新的改革以满足时代的需求。根据此问题,研发一套网上订餐系统,既能够大大提高信息的检索、变更与维护的工作效率,也能够方便信息系统的管理运用,从而减少信息管理成本,提高效率。 该网上订餐系统采用B/S架构、前后端分离以及B/C模型进行设计,并采用Java语言以及 SpringBoot框架进行开发。本系统主要设计并完成了用户注册、登录,购买美食过程、个人信息修改等,管理员对用户信息、美食分类、美食信息、系统信息、订单信息等功能,进行维护与管理。该系统操作简单,界面设计简单,不仅能基本满足目前网上网上订餐系统的日常管理工作,而且能有效降低人员成本和时间成本,为网上网上订餐系统工作提供方便。 关键词:网上订餐系统;B/S 架构;SpringBoot框架
内容概要:本文围绕电力系统状态估计中的异常检测与分类展开,重点介绍了基于Matlab实现的多种状态估计算法和技术,包括卡尔曼-加权最小二乘(KEWLS)、KEWLS-KF(KKF)以及粒子滤波等方法在电力系统中的应用。文档还涵盖了多源数据融合、信息融合与状态估计、观测信号含异常值的状态估计方法等内容,强调了异常数据的识别与处理对提升状态估计精度的重要性。此外,文中提到了多个相关研究方向的Matlab代码实现案例,如轴承故障诊断、永磁同步电机控制、信道估计、信号去噪等,展示了状态估计技术在不同工程领域的广泛应用。; 适合人群:具备一定电力系统、自动化或控制理论基础,熟悉Matlab/Simulink仿真环境,从事科研、教学或工程应用的研发人员、研究生及高【状态估计】电力系统状态估计中的异常检测与分类(Matlab代码实现)年级本科生;尤其适合开展状态估计、故障诊断、智能电网等相关课题研究的人员; 使用场景及目标:① 掌握电力系统中状态估计的基本原理与异常检测机制;② 学习并复现主流状态估计算法(如KEWLS、KKF、粒子滤波)的Matlab实现;③ 借助提供的代码资源开展科研仿真、论文复现或项目开发;④ 拓展多源信息融合与鲁棒估计方法的应用能力; 阅读建议:此资源以实际代码实现为核心,建议读者结合理论学习与编程实践,按目录顺序逐步深入,优先掌握状态估计基础模型后再进入异常分类与融合估计部分;同时可利用附带的网盘资源获取完整代码包,进行调试与二次开发,提升科研效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值