javascript中this指针探讨

 javascript是一门类java语言有很多跟java相类似的特点,但也仅是类似而已,真正使用中还是有很大的差别。this指针常常让很多初学者抓狂,本人也曾为此困惑不解,查找过很多资料,今天在这里总结一下,希望能帮助后来者更快驯服这只拦路虎。网上有很多讲解this指针的文章其中不乏精品,以我看来了解this指针关键在于掌握javascript中函数的四种调用模式。那么什么是调用?调用指的是跟在任何产生一个函数值的表达式之后使用"()",obj.f()这种方式成为调用,obj.f这种方式称为访问。

  一、方法调用模式:

  在该调用模式中函数作为一个对象的方法被调用:obj.fun()。当函数以此种形式被调用时this指针绑定到调用该方法的对象上即obj。

var myObj = {
  value: 0,
  increase: function(){
    this.value++;
    console.log(this.value);
  }
};
// this绑定到myObj对象上
myObj.increase(); // 1

myObj2 = {
  value: 10
};
myObj2.increase = myObj.increase;// myObj2与myObj的increase指向同一函数引用地址
// this绑定到myObj2对象上
myObj2.increase(); // 11

所以在将一个html元素的某一事件绑定某一函数时,若函数中使用this指针,则this指针会被绑定到该元素上。

var ele1 = document.getElementById('id');
ele1.addEventListener('click', myObj.increase, false);
// 该绑定相当于
ele1.onclick = myObj.increase;
// 事件触发时即相当于调用click方法
ele1.click(); // 与上文中myObj2.increase一个道理,若ele1中没有value属性则会报错

 二、函数调用模式:

  当函数没有当做方法调用即没有被一个对象通过点语法,这时它被当做一个函数来调用。以此模式调用函数时,this被绑定到全局对象。

var value = -10;
var myObj = {
  value: 0,
  increase: function(){
    this.value++;
    console.log(this.value);
  }
};
// this绑定到myObj对象上
myObj.increase(); // 1

var gInc = myObj.increase;
gInc(); // -9 this绑定到window对象上

所以在方法中使用内部函数时要特别注意,下例中other函数中的this并未绑定到myObj对象上

var value = -10;
var myObj = {
  value: 0,
  increase: function(){
    this.value++;
    console.log(this.value);
    var other = function(){
      this.value++;
      console.log(this.value);
    };
    other();// -9, 这时other中的this绑定到window对象上
  }
};
// this绑定到myObj对象上
myObj.increase(); // 1

var gInc = myObj.increase;
gInc(); // -8
///////////////
//1
//-9
//-8
//-7

幸运的是我们可以使用以下方式来在other中访问myObj对象:

var myObj = {
  value: 0,
  increase: function(){
    this.value++;
    console.log(this.value);
    var that = this;
    var other = function(){
      that.value++;
      console.log(that.value);
    };
    other();// -9, 这时other中的this绑定到window对象上
  }
};
// this绑定到myObj对象上
myObj.increase(); // 1
//////////////
//1
//2

  作为浏览器兼容性中很重要的一条:element.attachEvent绑定事件方法,当该事件触发时this指针并未绑定到element对象上,而是绑定到了window对象上,原因在于IE中事件触发时,响应函数是以一个独立函数即函数调用模式来调用的

 

  三、构造器模式调用:

  如果在一个函数前面加上new调用则成为构造器模式调用。使用new运算符,会产生一个连接到该函数prototype的新对象,this指针则被绑定到这个新对象上。

var P = function(n){
  this.name = n;
}
P.prototype.getName = function(){
  console.log(this.name);
}
var p = new P('woodtree'); // 这时P中的this对象呗绑定p指向的对象引用上
p.getName();// woodtree

同时new运算符还会改变函数的返回值。以函数调用模式调用P即P()返回undefined,而通过new运算符则返回一个新对象。new运算符类似于以下Function.prototype.create函数:

var P = function(n){
  this.name = n;
}
P.prototype.getName = function(){
  console.log(this.name);
}
var p = new P('woodtree');
p.getName();

Object.prototype.create = Object.create || function(proto){
  var F = function(){};
  F.prototype = proto;
  return new F();
}

Function.prototype.create = function(){
  var that = object.create(this.prototype);
  var obj = this.apply(that, arguments);

  return obj || that;
}
var p2 = P.create('hello new');
p2.getName(); // hello new

  四、apply、call模式调用:

  javascript中函数对象继承自Object亦可以拥有方法。call跟apply允许我们选择this的绑定对象。这两个方法的区别在于apply第二个参数必须是一个数组或者类数组对象即拥有length属性(arguments、NodeList、HTMLElementCollection等),而call除了第一个参数为要绑定this的对象外,后可跟无数的参数,参数之间用逗号间隔。

var P = function(n){
  this.name = n;
}
P.prototype.getNameAndAge = function(age){
  console.log(this.name + age);
}

P.prototype.getNameAndAge.call({
  name: 'catboat'
}, 99);
P.prototype.getNameAndAge.apply({
  name: 'lanuch'
}, [99]);

通过使用apply与call方法我们可以自行实现ES5中的bind函数

var P = function(n){
  this.name = n;
}
P.prototype.getNameAndAge = function(age, age2){
  console.log(this.name + age + age2);
}

P.prototype.getNameAndAge.call({
  name: 'catboat'
}, 99);
P.prototype.getNameAndAge.apply({
  name: 'lanuch'
}, [99]);

Function.prototype.bindContext = function(that){
  var _method = this,
  slice = Array.prototype.slice,
  args = slice.call(arguments, 1);

  return function(){
    _method.apply(that, Array.prototype.concat.apply(args, arguments));
  }
};

var f1 = P.prototype.getNameAndAge.bindContext({
  name: 'barque'
}, 88);
f1(9);

同样我们也可以解决IE中事件触发时,this指针问题(以下代码来自《Javascript框架设计》):

1 var addEvent = document.addEventListener ? function(el, type, fn, capture){
2   return el.addEventListener(type, fn, capture);
3 } : function(el, type, fn){
4   el.attachEvent('on' + type, fn.bindContext(el, event));
5 }
时间: 2024-08-04 09:07:01

javascript中this指针探讨的相关文章

探寻JavaScript中this指针指向_基础知识

探寻JavaScript中this指针指向 首先必须要说的是,this的指向在函数定义的时候是确定不了的,只有函数执行的时候才能确定this到底指向谁,实际上this的最终指向的是那个调用它的对象(这句话有些问题,后面会解释为什么会有问题,虽然网上大部分的文章都是这样说的,虽然在很多情况下那样去理解不会出什么问题,但是实际上那样理解是不准确的,所以在你理解this的时候会有种琢磨不透的感觉),那么接下来我会深入的探讨这个问题. 为什么要学习this?如果你学过函数式编程,面向对象编程,那你肯定知

对JavaScript中this指针的新理解分享_javascript技巧

一直以来对this的理解只在可以用,会用,却没有去深究其本质.这次,借着<JavaScript The Good Parts>,作了一次深刻的理解.(所有调试都可以在控制台中看到,浏览器F12键) 下面我们一起来看看这个this吧. 在我们声明一个函数时,每个函数除了有定义时的parameters(形参),自身还会有额外的两个参数,一个是this,一个是arguments(实参).arguments就是函数实际接受到的参数,是一个类数组.arguments我只做个简略的介绍,重点我们放在thi

Javascript中如何引用指针

本文介绍Javascript中引用指针的方法. 请尝试完成下列完形填空: /* 创建一个队列,头为head0,尾为tail0 */ function IntList(head0, tail0){ this.head = head0 || 0; this.tail = tail0 || null; } /* 返回一个IntList包含数组中的所有数 */ IntList.list = function(__args){ var sentinel = new IntList(), len = __a

深入探讨javascript中的数据类型_基础知识

学一门编程语言,无非两方面:一是语法,二是数据类型.类C语言的语法不外乎if.while.for.函数.算术运算等,面向对象的语言再加上object. 语法只是语言设计者预先做的一套规则,不同语言语法不尽相同,但都有一些共通点,对于熟悉一两门编程语言的人,学其他的编程语言时,语法往往不是问题(当然,如果你一直学的是类C语言,那么首次接触lisp时肯定也要花些时间),学习的重点往往是数据类型及其相关操作上,不是有句老话:"数据结构+算法=程序"!其次,有些语言的语法本身就存在设计问题(j

指针-Javascript中通过点操作符定义属性和字面量形式定义的区别

问题描述 Javascript中通过点操作符定义属性和字面量形式定义的区别 我已经有一种这样的困惑:第一种:function Person(){};Person.name = ""liang"";Person.age = 22 ;Person.sayName = function(){alert(this.name)}alert(Person.prototype.constructor == Person) ; // true在以上这个代码中,我创建了一个构造函数P

浅谈JavaScript中指针和地址_javascript技巧

个人理解:指针只是指向内存的一个索引:而地址则是内存中确切的位置. 下面是函数中关于指针和地址一个小例子: function sum(num1,num2){ return num1+num2; } alert(sum(10,10)); //20 var anotherSum=sum; alert(anotherSum(10,10)); //20 sum=null; alert(anotherSum(10,10)); //20  注意:使用不带圆括号的函数的名是访问函数指针,而非调用函数,所以 s

浅谈JavaScript中的this指针和引用知识_javascript技巧

this是javascript的一个关键字,随着函数使用场合不同,this的值会发生变化.但是总有一个原则,那就是this指的是调用函数的那个对象. this指针在传统OO语言中,是在类中声明的,表示对象本身.在JavaScript中,this表示当前上下文,即调用者的引用 ********this永远指向的是(函数对象)的所有者 this和全局对象: var a = 1; function foo(){ var b = 2; console.log(this.a+b);//3 } foo();

在JavaScript中实现类的方式探讨_javascript技巧

在 javascript 中有很多方式来创建对象,所以创建对象的方式使用起来非常灵活.那么,到底哪一种方式是最恰当的对象创建方式呢?构造模式,原型模式还是对象原意模式(Object literal)呢? 但这些模式具体是怎么回事呢? 在开始讲解之前,让我们先清楚地介绍一下关于 javascript 基本知识. 有没有可能在 javascript 中实现面向对象编程的方式呢? 答案是可能的,javascript 是可以创建对象的!这种对象可以包含数据及能够操作数据的方法,甚至可以包含其他对象.它没

JavaScript中对循环语句的优化技巧深入探讨_javascript技巧

循环是所有编程语言中最为重要的机制之一,几乎任何拥有实际意义的计算机程序(排序.查询等)都里不开循环. 而循环也正是程序优化中非常让人头疼的一环,我们往往需要不断去优化程序的复杂度,却因循环而纠结在时间复杂度和空间复杂度之间的抉择. 在 javascript 中,有3种原生循环,for () {}, while () {}和do {} while (),其中最为常用的要数for () {}. 然而for正是 javascript 工程师们在优化程序时最容易忽略的一种循环. 我们先来回顾一下for