【JavaScript】this 指向

1、this 指向谁

多数情况下,this 指向调用它所在方法的那个对象。即谁调的函数,this 就归谁。

当调用方法没有明确对象时,this 就指向全局对象。在浏览器中,指向 window;在 Node 中,指向 Global。(严格模式下,指向 undefined)

this 的指向是在调用时决定的,而不是在书写时决定的。这点和闭包恰恰相反。

2、区分 “声明位置” 与 “调用位置”

js 是词法作用域模型,对象或方法,其生命周期只和声明位置有关。

示例:

// 声明位置
var me = {
  name: 'xiaohong',
  hello: function() {
    console.log(`${this.name}`)
  }
}
var you = {
  name: 'xiaoming',
  hello: me.hello
}

// 调用位置
me.hello() 	// xiaohong
you.hello() // xiaoming

两次调用的 this 也就分别指向了 me 和 you。

// 声明位置
var me = {
  name: 'xiaohong',
  hello: function() {
    console.log(`${this.name}`)
  }
}
var name = 'xiaoming'
var hello = me.hello 
// 调用位置
me.hello() 	// xiaohong
hello() 		// xiaoming

me.hello 赋值给变量 hello 时,实际上是在创建一个新的函数引用,这个引用指向 me 对象中的 hello 方法。但是,这个新的引用(即变量 hello)并不保留与 me 对象的任何关联或绑定。

调用 hello() 时,是在全局作用域中调用这个函数,而不是作为 me 对象的方法调用。因此,在这个调用中,this 的值不会指向 me 对象。

// 声明位置
var me = {
  name: 'xiaohong',
  hello: function() {
    console.log(`${this.name}`)
  }
}
var you = {
  name: 'xiaoming',
  hello: function() {
    var targetFunc = me.hello
    targetFunc()
  }
}
var name = 'xiaosan'
// 调用位置
you.hello() // xiaosan

调用一个对象的方法时(例如 me.hello()),this 在该方法内部指向调用该方法的对象(在这个例子中是 me)。但是,将一个方法赋值给一个变量(例如 var targetFunc = me.hello),然后像普通函数那样调用这个变量(例如 targetFunc()),this 就不会再指向原来的对象了。相反,在非严格模式下,this 会默认指向全局对象(在浏览器中是 window),而在严格模式下,this 会是 undefined

3、普通函数的this指向总结

① 默认绑定:

  • 在全局中声明的变量和函数(默认指向Window);

  • 函数独立调用时(声明式函数、匿名函数 / 赋值式的方式、闭包)(都是指向Window);

  • 立即执行函数、 setTimeout、setInterval 指向 window

② 隐式绑定:

  • 对象调用(也就是谁调用就是指向谁,所以就是指向调用这个函数的对象)

(存在隐式丢失的问题:函数赋值和参数赋值的情况);

  • 绑定的事件处理函数(指向的是绑定事件处理函数的标签);

③ 显式绑定:

④ new绑定:

  • 构造函数中的this指向实例化出来的对象;

4、箭头函数

箭头函数内部是没有this指向的,箭头函数的this指向父级作用域的this;如果没有,则this指向的就是Window。

箭头函数注意事项:

  • 使用了箭头函数,this就不是指向window,而是父级(指向是可变的)

  • 不能够使用arguments对象

  • 不能用作构造函数,这就是说不能够使用new命令,否则会抛出一个错误

      var Foo = () => {
        console.log(this);
      };
      var a = new Foo(); // 报错Foo is not a constructor
    
  • 不可以使用 yield 命令,因此箭头函数不能用作 Generator 函数

示例:

function foo() {
    console.log(this) // obj对象
    // 情况一
    // function test() {
    //     console.log(this) // Window
    // }
    // test()

    // 情况二
    // function test() {
    //     console.log(this) // obj对象
    // }
    // test.call(this)

    // 情况三
    var test = () => {
        console.log(this) // obj对象
    }
    test()
}
var obj = {
    a: 1,
    foo: foo
}
obj.foo()
const obj = {
  sayThis: () => {
    console.log(this);
  }
};
// 因为JavaScript没有块作用域,所以在定义sayThis的时候,里面的this就绑到window上去了
obj.sayThis(); 	// window 
const globalSay = obj.sayThis;
globalSay(); 		// window 浏览器中的global对象
const liEle = document.querySelectorAll('li');
liEle.forEach((item, key) => {
    // 箭头函数的this指向的是父级程序
    // forEach()的this指向window
    console.log('打印1', this) // Window
 
    item.addEventListener('click', () => {
        console.log('打印2', this) // Window
    })
})
const button = document.getElementById('btn');
button.addEventListener('click', () => {
    console.log(this) // Window
})
function Cat(title) {
  this.title = title
}
Cat.prototype.sayName = () => {
  console.log(this) // Window
  return this.title
}
const cat = new Cat('我是标题啊');
console.log(cat.sayName()) // undefined
function foo() {
    console.log(this) // obj对象
    var test = () => {
        console.log(this) // obj对象
    }
    return test
}
var obj = {
    a: 1,
    foo: foo
}
obj.foo()()
function foo1() {
    console.log(this) // obj对象
}
var foo2 = () => {
    console.log(this) // Window
}
var obj = {
    a: 1,
    foo1: foo1,
    foo2: foo2
}
obj.foo1()
obj.foo2()
function foo1() {
    console.log(this) // obj2对象
}
var foo2 = () => {
    console.log(this) // Window
}
var obj = {
    a: 1,
    foo1: foo1,
    foo2: foo2
}
var obj2 = {
    a: 2
}
obj.foo1.call(obj2)
obj.foo2.call(obj2)
function foo() {
    console.log(this) // Window
    var test = () => {
        console.log(this) // Window
    }
    return test
}
var obj = {
    a: 1,
    foo: foo
}
var obj2 = {
    a: 2,
    foo: foo
}
foo().call(obj)
const obj = {
    // 普通函数:this指向调用它的对象
    fn1: function() {
        console.log(this) // {fn1: ƒ, fn2: ƒ, fn3: ƒ}
    },
 
    // 箭头函数:this指向是父级程序的this指向
    // 父级程序是obj对象,但只有函数有this,obj对象没有this
    // 父级程序没有this,指向的是window
    fn2: () => {
        console.log(this) // Window对象
    }, 
 
    // fn3是一个普通函数:this指向的是obj对象
    fn3: function() {
        // fn4是一个箭头函数:this指向的是父级程序的this指向
        // 父级程序是fn3,fn3的this指向的是obj对象,所以fn4箭头函数的this也是指向obj对象
        const fn4 = () => {
            console.log(this) // {fn1: ƒ, fn2: ƒ, fn3: ƒ}
        }
        fn4()
    }
}
obj.fn1()
obj.fn2()
obj.fn3()
var x = 11;
var obj = {
    x: 22,
    y: this,
    say: () => {
        console.log(this.x);
    }
}
obj.say();					// 11
console.log(obj.y);	// window

解析:obj 对象中的 this 指向的就是 window,也就是全局环境。所以obj.y打印的是window对象;**因为箭头函数中的 this 指向父级程序的指向。**从以上例子可以看出来,父级obj指向了window,所以this.x打印的是11

var a = 11;
function fn() {
  this.a = 22;
  let b = () => {
    console.log(this.a)
  }
  b();
}
var x = new fn();	// 22

解析:箭头函数中会往上寻找this,直到找到所代表的this为止。例子中,构造函数被实例化成为一个对象x,那x中的this指代的就是对象x本身,所以箭头函数this就代表x对象,x对象内部的a值为22,所以输出22。

var name = 'window'
var obj1 = {
    name: '1',
    fn1: function(){ 
        console.log(this.name) 
    },
    fn2: () => console.log(this.name),
    fn3: function (){
        return function(){
            console.log(this.name)
        }
    },
    fn4: function(){
        return () => console.log(this.name)
    }
}
var obj2 = {
    name: '2'
};

obj1.fn1();                 // 1
obj1.fn1.call(obj2);        // 2
obj1.fn2();                 // window
obj1.fn2.call(obj2);        // window
obj1.fn3();                 // window
obj1.fn3().call(obj2);      // 2
obj1.fn3.call(obj2)();      // window
obj1.fn4();                 // 1
obj1.fn4().call(obj2);      // 1
obj1.fn4.call(obj2)();      // 2
function Foo() {
  getName = function () {
    alert(1);
  };
  return this;
}
Foo.getName = function () {
  alert(2);
};
Foo.prototype.getName = function () {
  alert(3);
};
var getName = function () {
  alert(4);
};
function getName() {
  alert(5);
}

Foo.getName();						// 2
getName();								// 4
Foo().getName();					// 1
getName();								// 1

new Foo.getName();				// 2
new Foo().getName();			// 3
new new Foo().getName();	// 3

解析:

预编译中声明提升:function getName(){} 属于函数声明式,提升到最前面。

所以,全局的getName被替换成为function(){ alert(4) }

执行完Foo()后,函数返回了this,this指向window。所以Foo().getName()相当于window.getName(),而函数中的getName是全局的,执行函数的时候,替换掉了之前的输出为4的getName,当前就是输出1

9、资料

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值