react bind

前置知识

const test ={
        name:'dabao',
        getName:function(){
            console.log(this,this.name) //dabao
        }
    }

    test.getName()

    let temp = test.getName //此时的this指向替换为了window

    temp() //undefined

原因:在temp作为中间量转换的时候丢失了this的指向

在react中 bind的原理同上方一致,在jsx中传递的不是一个字符串而是一个函数,如(onClick={this.handleClick})
此时的onClick即为中间变量,会导致this值的丢失,需要重新绑定

js中绑定this值有以下几种方式

1 隐式绑定
2 显示绑定
3 new 绑定
4 window绑定
5 箭头函数绑定

this值因为上下文的转换导致其变化多端,下面从这5个方面详细分析一下

  • 隐式绑定
    最常见的称为隐式绑定
 const test = {
        name: 'dabao',
        getName: function() {
            console.log(this.name)
        }
    }

    test.getName() //dabao

稍微嵌套的代码

  const wrapper = {
        name: 'username1',
        getName: function() {
            console.log(this.name)  //username1
        },
        inner: {
            name: 'username2',
            getName: function() {
                console.log(this.name) //username2
            }
        }
    }

wrapper调用显示是username1,而wrapper.inner调用函数的时候,this值被绑定到inner了,所以显示username2

  • 显示绑定
    从代码入手
  function getname() {
        console.log(this.name)
    }

    const user = {
        name: 'dabao'
    }

    getname()

getname()的结果为空,因为此时调用相当于window.getname(),window 下并没有name这个变量,所以是空,
此时给window加上全局变量name,如下

function getname() {
        name = 'global' //隐式声明变量
        console.log(this.name)
    }

    // name = 'global'

    const user = {
        name: 'dabao'
    }

    getname()  //global

当window下有那么值就可以正常输出了
但是现在我们想输出user下的name应该怎么做呢?
此时需要使用显式样绑定的方法,将getname的this指向user,就可以用call,apply和bind这几种方法了
1)bind方法

 var newFn = getname.bind(user) 

    newFn()

onClick = {this.handleClick.bind(this)}
this.handleClick相当于getname user相当于this,但是bind会返回一个新的函数newFn,你必须调用它才能执行
render每次调用的时候都会返回一个新的函数newFn,虽然解决了tihs指向问题,但是会消耗更多性能
除了使用user内部的函数外,还可以传递其他的参数

   var other = {
        name: 'other',
        myFunc: function(...args) {
            console.log('我叫' + this.name, '我喜欢:', ...args)
        }
    }
    const mine = {
        name: 'dabao'
    }

    other.myFunc.call(mine, 'movie', 'wangzhe') //我叫dabao 我喜欢: movie wangzhe
    other.myFunc.apply(mine, ['movie', 'wangzhe']) //我叫dabao 我喜欢: movie wangzhe
    other.myFunc.bind(mine, 'movie', 'wangzhe')() //我叫dabao 我喜欢: movie wangzhe

2)apply和call也可以达到显示绑定的结果,
但是格式不一样
other.myFunc.call(mine, 'movie', 'wangzhe') ----------------- call传参 字符串
other.myFunc.apply(mine, ['movie', 'wangzhe']) ------------- apply传参 数组
other.myFunc.bind(mine, 'movie', 'wangzhe')() -------------- bind传参 字符串 必须得调用函数

你可能感兴趣的:(react bind)