javascript 之this指针-11
前言
在《javascript 之执行环境-08》文中说到,当JavaScript代码执行一段可执行代码时,会创建对应的执行上下文(execution context)。对于每个执行上下文,都有三个重要属性:
- 变量对象(Variable object,VO)
- 作用域链(Scope chain)
- this
JavaScript中的this跟其他语言有些不一样,比如Java .net语言中的this是在代码的执行阶段是不可变的,而JavaScript的this是在调用阶段进行绑定。也因为这一性质给了this很大的灵活性,即当函数在不同的调用方式下都可能会导致this的值不同;
定义
this 对象是在运行时基于函数的执行环境绑定的,跟函数的调用位置有关而不是声明的位置;可以理解为this是在函数调用阶段绑定,也就是执行上下文创建的阶段进行赋值,保存在变量对象中;
四种绑定规则
new 构造函数绑定,this指向新创建的对象
function createPerson(){
return new person();
}
function person() {
this.name = "Joel";
this.say=function(){
console.log(p)
console.log('hello' + this.name)
}
}
var p = new person();
p.say();
console.log(p.name);//Joel
console.log('开始')
var t=createPerson();
t.say();
不管是直接new 还是通过function createPerson 函数返回的对象,this 都是指向了新创建出来的对象;
显示绑定,this指向传进去的对象
//call() apply() bind() 显示绑定
window.color = "red";
var Obj = { color: "blue" };
function sayColor() {
console.log(this.color);
}
sayColor();// red this window
sayColor.call(this);//red
sayColor.call(window);//red
sayColor.call(Obj);// blue 把this指针改为对象Obj
sayColor.apply(Obj);//blue 把this指针改为对象Obj
sayColor.bind(Obj)();//blue 把this指针改为对象Obj
如果你把null/undefined作为this的绑定对象传入call/apply/bind,这些值在调用时会被忽略,实际用的是默认的绑定规则;
function foo() {
console.log(this.a)
}
var a=2;
foo.call(null);//
foo.call(undefined);//
隐士绑定
以对象的方法形式调用,this指向当前这个对象
function foo(){
console.log(this)//{a: 2, name: "Joel", foo: ƒ}
console.log(this.a)//
}
var obj={
a:2,
name:'Joel',
foo:foo
};
obj.foo();
这时this指向当前obj对象,但是如果换种写法会造成this 丢失问题。
function foo(){
console.log(this)//{a: 2, name: "Joel", foo: ƒ}
console.log(this.a)//
}
var obj={
a:2,
name:'Joel',
foo:foo
};
obj.foo();
//this 丢失的问题
var t= obj.foo;
t(); //window undefined
变量t此时保存的是函数的引用跟obj已经没有关系,所以此时this指向window。
默认绑定 严格模式下this 绑定到undefined,否则绑定到全局对象 window
function foo(){
console.log(this)//window
console.log(this.a)//Joel
}
var a='Joel';
foo();
//严格模式
function fo(){
'use strict' //严格模式
console.log(this)//undefined
console.log(this.b)//报错 Cannot read property 'b' of undefined
}
var b='Joel';
fo();
以上是基本的this绑定规则,其中new、显示绑定很容易判断,其中比较容易错的是容易把默认绑定误认为是隐士绑定 如匿名函数、闭包、函数当做参数等;
独立调用:this 指向window
var name='Joel',age=12;
function say(){
function say1(){
console.log(this.name);//window
function say2(){
name+='-l'
console.log(this.name);//window
}
say2()
}
say1();
}
say(); //匿名函数
(function(){
console.log(this.name)
})()
function 当做参数传递其实跟独立调用一样的原理
function foo() {
console.log(this)//window
console.log(this.a)//oops global
}
function doFoo(fn) {
console.log(this);//window
fn();//类似与 foo()
}
var obj = {
a: 2,
foo: foo
}
var a = 'oops global';
doFoo(obj.foo);
同理setTimeout 也是一样
var obj = {
a: 2,
foo: function() {
console.log(this);
},
foo2: function() {
console.log(this); //this 指向 obj
setTimeout(this.foo, 1000); // this 指向 window
}
}
var a = 'oops global';
obj.foo2();
闭包中的this
var name = "Joel";
var obj = {
name: "My object",
getName: function() {
// var that = this; // 将getNameFunc()的this保存在that变量中
return function() {
return this.name;
};
}
}
console.log(obj.getName()()); // "Joel"
这里的虽然是对象的方法形式调用obj.getName(),在getName中的this是指向obj,但是返回的匿名函数中的this 为什么是window呢?
把最后的一句拆成两个步骤执行:
var t=obj.getName();
t();
是不是有点像在独立调用呢?如果需要访问obj中的name,只需要把this对象缓存起来,在匿名函数中访问即可,把var that=this;去掉注释即可;
总结
- this 是变量对象的一个属性,是在调用时被绑定的,跟函数的调用位置有关而不是声明的位置;
- 找到调用位置,根据绑定规则来分析this 绑定;
- 默认绑定严格模式下this 绑定到undefined,否则绑定到全局对象 window;
思考题
var length = 5;
var obj = {
foo: function (fn) {
console.log(this.length); // this => obj
fn(); // this => window
arguments[0](); // this => arguments
var bar = arguments[0];
bar(); // this => window
},
length: 10
}
var fn = function () {
console.log(this.length);
}
obj.foo(fn);
//10, 5, 1, 5
javascript 之this指针-11的更多相关文章
- 图说js中的this——深入理解javascript中this指针
没搞错吧!js写了那么多年,this还是会搞错!没搞错,javascript就是回搞错! ………… 文章来源自——周陆军的个人网站:http://zhoulujun.cn/zhoulujun/html ...
- javascript中this指针
看完此片文章豁然开朗,非常感谢.javascript技术难点(三)之this.new.apply和call详解 下面说一说自己的理解: this指针总是指向调用他的对象,其实我更愿意理解为:this指 ...
- javascript中this指针探讨
javascript是一门类java语言有很多跟java相类似的特点,但也仅是类似而已,真正使用中还是有很大的差别.this指针常常让很多初学者抓狂,本人也曾为此困惑不解,查找过很多资料,今天在这里总 ...
- 原生JavaScript技巧大收集(11~20)-(终于又被我找到这篇文章了)
11.原生JavaScript加入收藏夹 function AddFavorite(sURL, sTitle) { try { window.external.addFavorite(sURL, sT ...
- JavaScript——对this指针的新理解
一直以来对this的理解只在可以用,会用,却没有去深究其本质.这次,借着<JavaScript The Good Parts>,作了一次深刻的理解.(所有调试都可以在控制台中看到,浏览器F ...
- javascript 误用this指针 的情况
理解了this指针后,我们再来看看一些很容易误用this指针的情况. 示例1——内联式绑定Dom元素的事件处理函数 <script type="text/javascript" ...
- javascript中this指针的认识
javascript中上下文环境就是this指针,即被调用函数所处的环境.这个上下文环境在大多数情况下指的是函数运行时封装这个函数的那个对象:当不通过任何对象单独调用一个函数时,上下文环境指的就是全局 ...
- Javascript图片预加载详解 分类: JavaScript HTML+CSS 2015-05-29 11:01 768人阅读 评论(0) 收藏
预加载图片是提高用户体验的一个很好方法.图片预先加载到浏览器中,访问者便可顺利地在你的网站上冲浪,并享受到极快的加载速度.这对图片画廊及图片占据很大比例的网站来说十分有利,它保证了图片快速.无缝地发布 ...
- JavaScript的this指针到底指向哪?
编程过程中,着实十分困扰this的指向性,经过查阅一番资料,终于搞清楚了,在这里总结一下,全文分为以下三个部分: 什么是this指针? this指针指向哪里? 何时使用this? 一 什么是this指 ...
随机推荐
- Spring MVC如何测试Controller(使用springmvc mock测试)
在springmvc中一般的测试用例都是测试service层,今天我来演示下如何使用springmvc mock直接测试controller层代码. 1.什么是mock测试? mock测试就是在测试过 ...
- Laravel学习笔记(二)
解决了类自动加载的问题,剩下的问题就是看文档了,laravel的官方文档虽然简单,但是却包含了很多基础知识,学习Laravel最好先看看官方文档,我感觉帮助很大,因为laravel框架的源码看起来并不 ...
- mybatis防止sql注入
SQL注入是一种代码注入技术,用于攻击数据驱动的应用,恶意的SQL语句被插入到执行的实体字段中(例如,为了转储数据库内容给攻击者).[摘自] SQL injection - Wikipedi ...
- 56、jsのBOM对象与DOM对象
javascript的Bom和Dom对象使我们学习的重点,这篇随笔可以重点阅读 一.BOM对象 1.window对象 所有浏览器都支持 window 对象.概念上讲.一个html文档对应一个windo ...
- 运算符关键字。数据区别大小写。日期范围。判空的两种写法。NOT IN的两种写法。IN范围可含NULL,但NOT IN值范围不能含NULL。
比较:>,<,=,>=,<=,<>(!=) 逻辑:AND,OR,NOT 范围:BETWEEN...AND... 范围:IN,NOT IN 判空:IS NULL, I ...
- JMeter脚本获取变量名、检验字符串值
说明: 脚本中获取变量值用vars.get("变量名"); 校验String类型的值使用String.equals("字符串值"),而不能用String==&q ...
- JAVA NIO学习一:NIO简介、NIO&IO的主要区别
在前面学习了IO之后,今天我们开始进入NIO学习环节,首先我们会NIO做一个简单的介绍,让大家认识NIO,然后会和IO进行一个对比认识进行区分.好了,下面我们就开始学习: 一.NIO简介 1.概述 从 ...
- KVO的内部实现以及使用
转载自:http://www.cocoachina.com/applenews/devnews/2014/0107/7667.html KVO是实现Cocoa Bindings的基础,它提供了一种 ...
- Noip2016换教室(期望+DP)
Description 题目链接:Luogu Solution 这题结合了DP和概率与期望,其实只要稍微知道什么是期望就可以了, 状态的构造很关键,\(F[i][j][0/1]\)表示已经到第\(i\ ...
- [置顶]
Xamarin android 调用Web Api(ListView使用远程数据)
xamarin android如何调用sqlserver 数据库呢(或者其他的),很多新手都会有这个疑问.xamarin android调用远程数据主要有两种方式: 在Android中保存数据或调用数 ...