javascript中的this作用域详解


Posted in Javascript onJuly 15, 2019

Javascript中this的指向一直是困扰我很久的问题,在使用中出错的机率也非常大。在面向对象语言中,它代表了当前对象的一个引用,而在js中却经常让我觉得混乱,它不是固定不变的,而是随着它的执行环境的改变而改变。

在Javascript中this总是指向调用它所在方法的对象。因为this是在函数运行时,自动生成的一个内部对象,只能在函数内部使用。

下面我们分几种情况深入分析this的用法:

1.全局的函数调用

function globalTest() {
    this.name = "global this";
    console.log(this.name);
  }
  globalTest(); //global this

以上代码中,globalTest()是全局性的方法,属于全局性调用,因此this就代表全局对象window。为了充分证明this是window,对代码做如下更改:

var name = "global this";

  function globalTest() {
    console.log(this.name);
  }
  globalTest(); //global this

name作为一个全局变量,运行结果仍然是“global this”,说明this指向的是window。在方法体中我们尝试更改全局name,再次调用方法输出“rename global this”, 说明全局的name在方法内部被更改。代码如下:

var name = "global this";

  function globalTest() {
    this.name = "rename global this"
    console.log(this.name);
  }
  globalTest(); //rename global this

根据以上三段代码,我们得出结论:对于全局的方法调用,this指向的是全局对象window,即调用方法所在的对象。

2.对象方法的调用

如果函数作为对象的方法调用,this指向的是这个上级对象,即调用方法的对象。 在以下代码中,this指向的是obj对象。

function showName() {
    console.log(this.name);
  }
  var obj = {};
  obj.name = "ooo";
  obj.show = showName;
  obj.show(); //ooo

3.构造函数的调用

构造函数中的this指向新创建的对象本身。

function showName() {
    this.name = "showName function";
  }
  var obj = new showName();
  console.log(obj.name); //showName function

上述代码中,我们通过new关键字创建一个对象的实例,new关键字可以改变this的指向,将这个this指向对象obj。
我们再增加一个全局的name,用以证明this指向的不是global:

var name = "global name";

  function showName() {
    this.name = "showName function";
  }
  var obj = new showName();

  console.log(obj.name); //showName function
  console.log(name); //global name

在构造函数的内部,我们对this.name进行赋值,但并没有改变全局变量name。

4.apply/call调用时的this

apply和call都是为了改变函数体内部的this指向。 其具体的定义如下:

call方法:

语法:call(thisObj,Object)

定义:调用一个对象的一个方法,以另一个对象替换当前对象。

说明:

call 方法可以用来代替另一个对象调用一个方法。call 方法可将一个函数的对象上下文从初始的上下文改变为由 thisObj 指定的新对象。

如果没有提供 thisObj 参数,那么 Global 对象被用作 thisObj。

apply方法:

语法:apply(thisObj,[argArray])

定义:应用某一对象的一个方法,用另一个对象替换当前对象。

说明:

如果 argArray 不是一个有效的数组或者不是 arguments 对象,那么将导致一个 TypeError。

如果没有提供 argArray 和 thisObj 任何一个参数,那么 Global 对象将被用作 thisObj, 并且无法被传递任何参数。

var value = "Global value";

  function FunA() {
    this.value = "AAA";
  }

  function FunB() {
    console.log(this.value);
  }
  FunB(); //Global value 因为是在全局中调用的FunB(),this.value指向全局的value
  FunB.call(window); //Global value,this指向window对象,因此this.value指向全局的value
  FunB.call(new FunA()); //AAA, this指向参数new FunA(),即FunA对象

  FunB.apply(window); //Global value
  FunB.apply(new FunA()); //AAA

在上述代码中,this的指向在call和apply中是一致的,只不过是调用参数的形式不一样。call是一个一个调用参数,而apply是调用一个数组。具体的会在以后的博文中单独阐述。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
用javascript操作xml
Nov 04 Javascript
JQuery之拖拽插件实现代码
Apr 14 Javascript
JavaScript动态加载样式表的方法
Mar 21 Javascript
Bootstrap每天必学之附加导航(Affix)插件
Apr 25 Javascript
js替换字符串中所有指定的字符(实现代码)
Aug 17 Javascript
基于JS快速实现导航下拉菜单动画效果附源码下载
Oct 27 Javascript
javascript基于原型链的继承及call和apply函数用法分析
Dec 15 Javascript
ES6新数据结构Map功能与用法示例
Mar 31 Javascript
Vue-Quill-Editor富文本编辑器的使用教程
Sep 21 Javascript
详解Vue前端生产环境发布配置实战篇
May 07 Javascript
Vue调用后端java接口的实例代码
Oct 28 Javascript
基于JS实现快速读取TXT文件
Aug 25 Javascript
微信小程序页面上下滚动效果
Nov 18 #Javascript
node.js实现上传文件功能
Jul 15 #Javascript
js canvas实现5张图片合成一张图片
Jul 15 #Javascript
js使用文件流下载csv文件的实现方法
Jul 15 #Javascript
基于Taro的微信小程序模板消息-获取formId功能模块封装实践
Jul 15 #Javascript
微信小程序开发技巧汇总
Jul 15 #Javascript
微信小程序中如何计算距离某个节日还有多少天
Jul 15 #Javascript
You might like
PHP 正则表达式之正则处理函数小结(preg_match,preg_match_all,preg_replace,preg_split)
2012/10/05 PHP
PHP和javascript常用正则表达式及用法实例
2014/07/01 PHP
PHP用户验证和标签推荐的简单使用
2016/10/31 PHP
yii2.0整合阿里云oss上传单个文件的示例
2017/09/19 PHP
TP3.2.3框架使用CKeditor编辑器在页面中上传图片的方法分析
2019/12/31 PHP
javascript 程序库的比较(一)之DOM功能
2010/04/07 Javascript
Jquery+ajax请求data显示在GridView上(asp.net)
2010/08/27 Javascript
js动态创建及移除div的方法
2015/06/03 Javascript
使用javascript将时间转换成今天,昨天,前天等格式
2015/06/25 Javascript
深入解析Backbone.js框架的依赖库Underscore.js的作用
2016/05/07 Javascript
AngularJS 模型详细介绍及实例代码
2016/07/27 Javascript
AngularJS下对数组的对比分析
2016/08/24 Javascript
在网页中插入百度地图的步骤详解
2016/12/02 Javascript
纯javaScript、jQuery实现个性化图片轮播【推荐】
2017/01/08 Javascript
javascript ES6 新增了let命令使用介绍
2017/07/07 Javascript
js中时间格式化的几种方法
2018/07/22 Javascript
JQuery Ajax动态加载Table数据的实例讲解
2018/08/09 jQuery
vue+echarts实现动态绘制图表及异步加载数据的方法
2018/10/17 Javascript
微信小程序实现简易table表格
2020/06/19 Javascript
最简单的vue消息提示全局组件的方法
2019/06/16 Javascript
[01:27]2014DOTA2展望TI 剑指西雅图IG战队专访
2014/06/30 DOTA
python socket网络编程步骤详解(socket套接字使用)
2013/12/06 Python
Python提示[Errno 32]Broken pipe导致线程crash错误解决方法
2014/11/19 Python
Python3如何解决字符编码问题详解
2017/04/23 Python
Python模块结构与布局操作方法实例分析
2017/07/24 Python
pandas 实现将重复表格去重,并重新转换为表格的方法
2018/04/18 Python
python 插入日期数据到Oracle实例
2020/03/02 Python
幼儿园家长会邀请函
2014/01/15 职场文书
本科生导师推荐信范文
2014/05/18 职场文书
女生节标语
2014/06/26 职场文书
幼儿园安全责任书范本
2014/07/24 职场文书
委托培训协议书
2014/11/17 职场文书
监理中标通知书
2015/04/16 职场文书
2016先进集体事迹材料范文
2016/02/25 职场文书
在K8s上部署Redis集群的方法步骤
2021/04/27 Redis
使用kubeadm命令行工具创建kubernetes集群
2022/03/31 Servers