This in JavaScript
声明 本文仅为读书笔记并致力于理解Js中的this关键字。如有雷同,纯属巧合。原因有二,其一:有幸看了同一本书,其二:this这玩意已被说烂了~
为什么要用this?
this提供了优雅的方式隐式传递对象的引用
上实例:
function sayEng(){
console.log(this.name+":sayEng");
}
function sayChi(){
console.log(this.name+":sayChi");
}
function stu(){
name = "stu";
}
function tea(){
name = "tea";
}
sayEng.call(stu);//stu:sayEng
sayEng.call(tea);//tea:sayEng
sayChi.call(stu);//stu:sayChi
sayChi.call(tea);//tea:sayChi
当不使用this时
function sayEng(ctx){
console.log(ctx.name+"sayEng");
}
function sayChi(ctx){
console.log(ctx.name+"sayChi");
}
function stu(){
name = "stu";
}
function tea(){
name = "tea";
}
sayEng(stu);//stu:sayEng
sayEng(tea);//tea:sayEng
sayChi(stu);//stu:sayChi
sayChi(tea);//stea:sayChi
```
两种对比,可以看出当不使用this时,需**手动显式传递一个上下文对象**。而使用了this可以使得代码很精炼。而且传递上下文的方式会使得代码越来越混乱。
2. 对this的误解
1)指向自身(书中代码)
```javascript
function foo(num) {
console.log( "foo: " + num );
// count用于记录foo的被调用次数
this.count++;
}
foo.count = 0;
var i;
for (i=0; i<10; i++) {
if (i > 5) {
foo( i );
}
}
console.log( foo.count ); // 0 -- ??
```
**为什么是0,而不是4?原因在于,此时调用foo函数的是window,this的指向是window。**
解决办法有两种:
其一,用foo代替this。也就是foo.count++。
其二,强制this指向foo,通过call,或者apply。也就是在if里面改为foo.call(foo,i)。
2)指向函数的作用域
```javascript
function foo() {
var a = 2;
this.bar();
}
function bar() {
console.log( this.a );
}
foo(); //ReferenceError:a is not defined
```
**原因在于:调用foo的对象为window**
3. 绑定规则
1)默认绑定
若函数是直接使用不带任何修饰的函数引用的,它便是应用了默认绑定。this指向window,上面的实例可以佐证。
p.s *当使用严格模式时,则不能将全局对象用于默认绑定*
2)隐式绑定
```javascript
function foo() {
console.log( this.a );
}
var obj = {
a: 2,
foo: foo
};
obj.foo(); // 2
```
本例就使用了默认绑定,this绑定在obj上。**另外,对象属性引用链只有上一层或者说最后一层在调用位置中起作用。**也即,obj1.obj2.foo(),this绑定在obj2中。
但是,隐式绑定经常会有一个问题产生--**隐式丢失**
```javascript
function foo() {
console.log( this.a );
}
var obj = {
a: 2,
foo: foo
};
var bar = obj.foo; // 函数别名
var a = "oops,global";
bar(); // "oops,globa"
```
this的指向不再是obj,而变成了window。丢失了this的指向。同理还有当obj.foo作为参数传递时丢失,原因在于参数传递相当于两个步骤,一赋值,二参数传递。
3)显示传递
通过call、apply的方式,将this绑定在对象上。具体call、apply的使用方法见本人博客[Call and Apply in JavaScript](http://www.cnblogs.com/zhaoww/p/5247827.html)
4)new绑定
```javascript
function foo(a) {
this.a = a;
}
var bar = new foo( 2 );
console.log( bar.a );// 2
```
使用new调用时,会构造一个对象并将this指向进行绑定。如本例的foo上。
4. 优先级
new绑定>显示绑定>隐式绑定>默认绑定
----------
>*本人较懒,部分代码原封不动摘抄自原著。还请见谅*
This in JavaScript的更多相关文章
- JavaScript之父Brendan Eich,Clojure 创建者Rich Hickey,Python创建者Van Rossum等编程大牛对程序员的职业建议
软件开发是现时很火的职业.据美国劳动局发布的一项统计数据显示,从2014年至2024年,美国就业市场对开发人员的需求量将增长17%,而这个增长率比起所有职业的平均需求量高出了7%.很多人年轻人会选择编 ...
- javascript中的Array对象 —— 数组的合并、转换、迭代、排序、堆栈
Array 是javascript中经常用到的数据类型.javascript 的数组其他语言中数组的最大的区别是其每个数组项都可以保存任何类型的数据.本文主要讨论javascript中数组的声明.转换 ...
- Javascript 的执行环境(execution context)和作用域(scope)及垃圾回收
执行环境有全局执行环境和函数执行环境之分,每次进入一个新执行环境,都会创建一个搜索变量和函数的作用域链.函数的局部环境不仅有权访问函数作用于中的变量,而且可以访问其外部环境,直到全局环境.全局执行环境 ...
- 探究javascript对象和数组的异同,及函数变量缓存技巧
javascript中最经典也最受非议的一句话就是:javascript中一切皆是对象.这篇重点要提到的,就是任何jser都不陌生的Object和Array. 有段时间曾经很诧异,到底两种数据类型用来 ...
- 读书笔记:JavaScript DOM 编程艺术(第二版)
读完还是能学到很多的基础知识,这里记录下,方便回顾与及时查阅. 内容也有自己的一些补充. JavaScript DOM 编程艺术(第二版) 1.JavaScript简史 JavaScript由Nets ...
- 《Web 前端面试指南》1、JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
- JavaScript权威指南 - 函数
函数本身就是一段JavaScript代码,定义一次但可能被调用任意次.如果函数挂载在一个对象上,作为对象的一个属性,通常这种函数被称作对象的方法.用于初始化一个新创建的对象的函数被称作构造函数. 相对 ...
- JavaScript自定义浏览器滚动条兼容IE、 火狐和chrome
今天为大家分享一下我自己制作的浏览器滚动条,我们知道用css来自定义滚动条也是挺好的方式,css虽然能够改变chrome浏览器的滚动条样式可以自定义,css也能够改变IE浏览器滚动条的颜色.但是css ...
- JavaScript进阶之路(一)初学者的开始
一:写在前面的问题和话 一个javascript初学者的进阶之路! 背景:3年后端(ASP.NET)工作经验,javascript水平一般般,前端水平一般般.学习资料:犀牛书. 如有误导,或者错误的地 ...
- 梅须逊雪三分白,雪却输梅一段香——CSS动画与JavaScript动画
CSS动画并不是绝对比JavaScript动画性能更优越,开源动画库Velocity.js等就展现了强劲的性能. 一.两者的主要区别 先开门见山的说说两者之间的区别. 1)CSS动画: 基于CSS的动 ...
随机推荐
- 洛谷P1738 洛谷的文件夹
原题目:点我 题目是一个略水的题,我机制地用面向对象做了...所以代码量急剧加大,100行233 模拟即可,字符串处理麻烦点.如果没有找到子文件夹就新建文件夹,如果有就进入该文件夹. 提示:高能,指针 ...
- 0801 am使用tp框架对数据库增删改查
增添数据,3种方法 function Text3() { $m=D("info"); //1.使用数组 $attr = array( "code"=>&q ...
- (Python )模块、包
本节开始学习模块的相关知识,主要包括模块的编译,模块的搜索路径.包等知识 1.模块 如果我们直接在解释器中编写python,当我们关掉解释器后,再进去.我们之前编写的代码都丢失了.因此,我们需要将我们 ...
- [VBS脚本]定时提示并关闭计算机
一.Natalia.bat 双击这个文件可以运行Natalia 控制台在打开后会变成最小化,标题为PowerAssistant,黑色背景淡绿色文字 运行脚本Natalia.vbs @ECHO OFF ...
- Unity5的AssetBundle的一点使用心得
昨天一位朋友在我这里留言,想让我写点Unity5的AssetBundle心得.于是我就看了相关的介绍,和自己确切的做了一次.下面来谈谈所谓的心得. 如果你觉得自己对AssetBundle不熟悉,建议先 ...
- POJ 3468 A Simple Problem with Integers (线段树)
题意:给定两种操作,一种是区间都加上一个数,另一个查询区间和. 析:水题,线段树. 代码如下: #pragma comment(linker, "/STACK:1024000000,1024 ...
- 评分视图的封装 (星星 RatingView)
#import "RatingView.h" #define kRatingScale 10 @implementation RatingView { UIView *_grayS ...
- Android Studio签名打包的两种方式
签名打包的两种方式: 注:给我们自己开发的app签名,就代表着我自己的版权,以后要进行升级,也必须要使用相同的签名才行.签名就代表着自己的身份(即keystore),多个app可以使用同一个签名. 如 ...
- 新浪微博SDK的使用
花了两天时间研究了一下新浪微博SDK,遇到了不少问题,有必要整理一下 1.首先下载下weiboSdk,导入weiboSDKD和weiboSDKDemo两个项目,这时发现导入的weiboSDKDemo项 ...
- java抓取网页数据,登录之后抓取数据。
最近做了一个从网络上抓取数据的一个小程序.主要关于信贷方面,收集的一些黑名单网站,从该网站上抓取到自己系统中. 也找了一些资料,觉得没有一个很好的,全面的例子.因此在这里做个笔记提醒自己. 首先需要一 ...