【源码阅读】Vue2中this的秘密

179 阅读2分钟

写过vue2项目的小伙伴都知道,调用数据时和调用方法时用this.xx就可以,那么你有没有好奇这是怎么实现的,如果是你,你准备怎么实现

辅助文章《前端容易忽略的 debugger 调试技巧》

我先搭建项目环境,建立一个页面index.html,通过CDN引入vue@2.6.14就可以了,既然猜不到怎么写的,那么就开卷有益了,直接进入源码内部,看看到底是怎么实现的,都用到了哪些知识

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta http-equiv="X-UA-Compatible" content="IE=edge" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>vue2-this</title>
  </head>
  <body>
    <script src="https://unpkg.com/vue@2.6.14/dist/vue.js"></script>
    <script>
      const vue2 = new Vue({
        data() {
          return {
            name: 'jack',
          }
        },
        methods: {
          showName() {
            return this.name
          },
        },
      })
      console.log(vue2.name)
      console.log(vue2.showName())
    </script>
  </body>
</html>

在new Vue处打断点,再通过F11一步一步进入内部,F11是单步执行下一个函数的意思

进入vue的构造函数,我们的应用就是从new Vue开始的

image.png

初始化函数 _init再进入这个函数内看看 F11进入函数,找到initState函数,根据语义猜到这里是初始Vue实例状态函数

image.png

initState函数内部逻辑,我们可以看到所有我们声明在Vue实例中的属性,包括函数, 例如:data,methods,watch,props等,继续看看this.xx形式取data中数据是怎么实现的,技巧是在要看的函数打断点,F11是进入函数内部,F8是切换各个断点

image.png

我们在initData处打个断点,然后F8切换到initData这里,F11进入这个函数内部,所有逻辑都在下面

image.png

注意这里的proxy函数,秘密就在这里,this.xx 其实就是this._data.xx 还是打断点找到它

image.png

同样的方法去看initMethods函数

总结

  • 看源码的基本技巧,掌握F8 F11 哪里不会就“断”哪里
  • call apply bind的使用技巧,业务代码中很少用到,看源码可没少遇到
  • 还有一个小问题,就是this的指向问题,有时间在总结总结