当学习成为了习惯,常识也就变成了常识。 感激各位的 点赞珍藏评论

新视频和文章会第一工夫在微信公众号发送,欢送关注:李永宁lyn

文章已收录到 github 仓库 liyongning/blog,欢送 Watch 和 Star。

指标

深刻了解 Vue 的初始化过程,再也不怕 面试官 的那道面试题:new Vue(options) 产生了什么?

找入口

想晓得 new Vue(options) 都做了什么,就得先找到 Vue 的构造函数是在哪申明的,有两个方法:

  • 从 rollup 配置文件中找到编译的入口,而后一步步找到 Vue 构造函数,这种形式 吃力
  • 通过编写示例代码,而后打断点的形式,一步到位,简略

咱们就采纳第二种形式,写示例,打断点,一步到位。

  • /examples 目录下减少一个示例文件 —— test.html,在文件中增加如下内容:
<!DOCTYPE html><html lang="en"><head>  <meta charset="UTF-8">  <title>Vue 源码解读</title></head><body>  <div id="app">    {{ msg }}  </div>  <script src="../dist/vue.js"></script>  <script>    debugger    new Vue({      el: '#app',      data: {        msg: 'hello vue'      }    })  </script></body></html>
  • 在浏览器中关上控制台,而后关上 test.html,则会进入断点调试,而后找到 Vue 构造函数所在的文件

点击查看演示动图,动图地址:https://p1-juejin.byteimg.com...

失去 Vue 构造函数在 /src/core/instance/index.js 文件中,接下来正式开始源码浏览,带着指标去浏览。

在浏览过程中如遇到看不明确的中央,可通过编写示例代码,而后应用浏览器的调试性能进行一步步调试,配合了解,如果还是了解不了,就做个备注持续向后看,兴许你看到其它中央,就忽然明确这个中央在做什么,或者回头再来看,就会懂了,源码这个货色,肯定要多看,要想精通,一遍两遍必定是不够的

源码解读 —— Vue 初始化过程

Vue

/src/core/instance/index.js
import { initMixin } from './init'// Vue 构造函数function Vue (options) {  // 调用 Vue.prototype._init 办法,该办法是在 initMixin 中定义的  this._init(options)}// 定义 Vue.prototype._init 办法initMixin(Vue)export default Vue

Vue.prototype._init

/src/core/instance/init.js
/** * 定义 Vue.prototype._init 办法  * @param {*} Vue Vue 构造函数 */export function initMixin (Vue: Class<Component>) {  // 负责 Vue 的初始化过程  Vue.prototype._init = function (options?: Object) {    // vue 实例    const vm: Component = this    // 每个 vue 实例都有一个 _uid,并且是顺次递增的    vm._uid = uid++    // a flag to avoid this being observed    vm._isVue = true    // 解决组件配置项    if (options && options._isComponent) {      /**       * 每个子组件初始化时走这里,这里只做了一些性能优化       * 将组件配置对象上的一些深层次属性放到 vm.$options 选项中,以进步代码的执行效率       */      initInternalComponent(vm, options)    } else {      /**       * 初始化根组件时走这里,合并 Vue 的全局配置到根组件的部分配置,比方 Vue.component 注册的全局组件会合并到 根实例的 components 选项中       * 至于每个子组件的选项合并则产生在两个中央:       *   1、Vue.component 办法注册的全局组件在注册时做了选项合并       *   2、{ components: { xx } } 形式注册的部分组件在执行编译器生成的 render 函数时做了选项合并,包含根组件中的 components 配置       */      vm.$options = mergeOptions(        resolveConstructorOptions(vm.constructor),        options || {},        vm      )    }    /* istanbul ignore else */    if (process.env.NODE_ENV !== 'production') {      // 设置代理,将 vm 实例上的属性代理到 vm._renderProxy      initProxy(vm)    } else {      vm._renderProxy = vm    }    // expose real self    vm._self = vm    // 初始化组件实例关系属性,比方 $parent、$children、$root、$refs 等    initLifecycle(vm)    /**     * 初始化自定义事件,这里须要留神一点,所以咱们在 <comp @click="handleClick" /> 上注册的事件,监听者不是父组件,     * 而是子组件自身,也就是说事件的派发和监听者都是子组件自身,和父组件无关     */    initEvents(vm)    // 解析组件的插槽信息,失去 vm.$slot,解决渲染函数,失去 vm.$createElement 办法,即 h 函数    initRender(vm)    // 调用 beforeCreate 钩子函数    callHook(vm, 'beforeCreate')    // 初始化组件的 inject 配置项,失去 result[key] = val 模式的配置对象,而后对后果数据进行响应式解决,并代理每个 key 到 vm 实例    initInjections(vm) // resolve injections before data/props    // 数据响应式的重点,解决 props、methods、data、computed、watch    initState(vm)    // 解析组件配置项上的 provide 对象,将其挂载到 vm._provided 属性上    initProvide(vm) // resolve provide after data/props    // 调用 created 钩子函数    callHook(vm, 'created')    // 如果发现配置项上有 el 选项,则主动调用 $mount 办法,也就是说有了 el 选项,就不须要再手动调用 $mount,反之,没有 el 则必须手动调用 $mount    if (vm.$options.el) {      // 调用 $mount 办法,进入挂载阶段      vm.$mount(vm.$options.el)    }  }}

resolveConstructorOptions

/src/core/instance/init.js
/** * 从组件构造函数中解析配置对象 options,并合并基类选项 * @param {*} Ctor  * @returns  */export function resolveConstructorOptions (Ctor: Class<Component>) {  // 配置我的项目  let options = Ctor.options  if (Ctor.super) {    // 存在基类,递归解析基类构造函数的选项    const superOptions = resolveConstructorOptions(Ctor.super)    const cachedSuperOptions = Ctor.superOptions    if (superOptions !== cachedSuperOptions) {      // 阐明基类构造函数选项曾经产生扭转,须要从新设置      Ctor.superOptions = superOptions      // 查看 Ctor.options 上是否有任何前期批改/附加的选项(#4976)      const modifiedOptions = resolveModifiedOptions(Ctor)      // 如果存在被批改或减少的选项,则合并两个选项      if (modifiedOptions) {        extend(Ctor.extendOptions, modifiedOptions)      }      // 选项合并,将合并后果赋值为 Ctor.options      options = Ctor.options = mergeOptions(superOptions, Ctor.extendOptions)      if (options.name) {        options.components[options.name] = Ctor      }    }  }  return options}

resolveModifiedOptions

/src/core/instance/init.js
/** * 解析构造函数选项中后续被批改或者减少的选项 */function resolveModifiedOptions (Ctor: Class<Component>): ?Object {  let modified  // 构造函数选项  const latest = Ctor.options  // 密封的构造函数选项,备份  const sealed = Ctor.sealedOptions  // 比照两个选项,记录不统一的选项  for (const key in latest) {    if (latest[key] !== sealed[key]) {      if (!modified) modified = {}      modified[key] = latest[key]    }  }  return modified}

mergeOptions

/src/core/util/options.js
/** * 合并两个选项,呈现雷同配置项时,子选项会笼罩父选项的配置 */export function mergeOptions (  parent: Object,  child: Object,  vm?: Component): Object {  if (process.env.NODE_ENV !== 'production') {    checkComponents(child)  }  if (typeof child === 'function') {    child = child.options  }  // 标准化 props、inject、directive 选项,不便后续程序的解决  normalizeProps(child, vm)  normalizeInject(child, vm)  normalizeDirectives(child)  // 解决原始 child 对象上的 extends 和 mixins,别离执行 mergeOptions,将这些继承而来的选项合并到 parent  // mergeOptions 解决过的对象会含有 _base 属性  if (!child._base) {    if (child.extends) {      parent = mergeOptions(parent, child.extends, vm)    }    if (child.mixins) {      for (let i = 0, l = child.mixins.length; i < l; i++) {        parent = mergeOptions(parent, child.mixins[i], vm)      }    }  }  const options = {}  let key  // 遍历 父选项  for (key in parent) {    mergeField(key)  }  // 遍历 子选项,如果父选项不存在该配置,则合并,否则跳过,因为父子领有同一个属性的状况在下面解决父选项时曾经解决过了,用的子选项的值  for (key in child) {    if (!hasOwn(parent, key)) {      mergeField(key)    }  }  // 合并选项,childVal 优先级高于 parentVal  function mergeField (key) {    // strats = Object.create(null)    const strat = strats[key] || defaultStrat    // 值为如果 childVal 存在则优先应用 childVal,否则应用 parentVal    options[key] = strat(parent[key], child[key], vm, key)  }  return options}

initInjections

/src/core/instance/inject.js
/** * 初始化 inject 配置项 *   1、失去 result[key] = val *   2、对后果数据进行响应式解决,代理每个 key 到 vm 实例 */export function initInjections (vm: Component) {  // 解析 inject 配置项,而后从祖代组件的配置中找到 配置项中每一个 key 对应的 val,最初失去 result[key] = val 的后果  const result = resolveInject(vm.$options.inject, vm)  // 对 result 做 数据响应式解决,也有代理 inject 配置中每个 key 到 vm 实例的作用。  // 不不倡议在子组件去更改这些数据,因为一旦祖代组件中 注入的 provide 产生更改,你在组件中做的更改就会被笼罩  if (result) {    toggleObserving(false)    Object.keys(result).forEach(key => {      /* istanbul ignore else */      if (process.env.NODE_ENV !== 'production') {        defineReactive(vm, key, result[key], () => {          warn(            `Avoid mutating an injected value directly since the changes will be ` +            `overwritten whenever the provided component re-renders. ` +            `injection being mutated: "${key}"`,            vm          )        })      } else {        defineReactive(vm, key, result[key])      }    })    toggleObserving(true)  }}

resolveInject

/src/core/instance/inject.js
/** * 解析 inject 配置项,从祖代组件的 provide 配置中找到 key 对应的值,否则用 默认值,最初失去 result[key] = val * inject 对象必定是以下这个构造,因为在 合并 选项时对组件配置对象做了标准化解决 * @param {*} inject = { *  key: { *    from: provideKey, *    default: xx *  } * } */export function resolveInject (inject: any, vm: Component): ?Object {  if (inject) {    // inject is :any because flow is not smart enough to figure out cached    const result = Object.create(null)    // inject 配置项的所有的 key    const keys = hasSymbol      ? Reflect.ownKeys(inject)      : Object.keys(inject)    // 遍历 key    for (let i = 0; i < keys.length; i++) {      const key = keys[i]      // 跳过 __ob__ 对象      // #6574 in case the inject object is observed...      if (key === '__ob__') continue      // 拿到 provide 中对应的 key      const provideKey = inject[key].from      let source = vm      // 遍历所有的祖代组件,直到 根组件,找到 provide 中对应 key 的值,最初失去 result[key] = provide[provideKey]      while (source) {        if (source._provided && hasOwn(source._provided, provideKey)) {          result[key] = source._provided[provideKey]          break        }        source = source.$parent      }      // 如果上一个循环未找到,则采纳 inject[key].default,如果没有设置 default 值,则抛出谬误      if (!source) {        if ('default' in inject[key]) {          const provideDefault = inject[key].default          result[key] = typeof provideDefault === 'function'            ? provideDefault.call(vm)            : provideDefault        } else if (process.env.NODE_ENV !== 'production') {          warn(`Injection "${key}" not found`, vm)        }      }    }    return result  }}

initProvide

/src/core/instance/inject.js
/** * 解析组件配置项上的 provide 对象,将其挂载到 vm._provided 属性上  */export function initProvide (vm: Component) {  const provide = vm.$options.provide  if (provide) {    vm._provided = typeof provide === 'function'      ? provide.call(vm)      : provide  }}

总结

Vue 的初始化过程(new Vue(options))都做了什么?

  • 解决组件配置项

    • 初始化根组件时进行了选项合并操作,将全局配置合并到根组件的部分配置上
    • 初始化每个子组件时做了一些性能优化,将组件配置对象上的一些深层次属性放到 vm.$options 选项中,以进步代码的执行效率
  • 初始化组件实例的关系属性,比方 &dollar;parent、&dollar;children、&dollar;root、&dollar;refs 等
  • 解决自定义事件
  • 调用 beforeCreate 钩子函数
  • 初始化组件的 inject 配置项,失去 ret[key] = val 模式的配置对象,而后对该配置对象进行浅层的响应式解决(只解决了对象第一层数据),并代理每个 key 到 vm 实例上
  • 数据响应式,解决 props、methods、data、computed、watch 等选项
  • 解析组件配置项上的 provide 对象,将其挂载到 vm._provided 属性上
  • 调用 created 钩子函数
  • 如果发现配置项上有 el 选项,则主动调用 &dollar;mount 办法,也就是说有了 el 选项,就不须要再手动调用 &dollar;mount 办法,反之,没提供 el 选项则必须调用 &dollar;mount
  • 接下来则进入挂载阶段

链接

  • 配套视频,关注微信公众号回复:"精通 Vue 技术栈源码原理视频版" 获取
  • 精通 Vue 技术栈源码原理 专栏
  • github 仓库 liyongning/Vue 欢送 Star

感激各位的:点赞珍藏评论,咱们下期见。


当学习成为了习惯,常识也就变成了常识。 感激各位的 点赞珍藏评论

新视频和文章会第一工夫在微信公众号发送,欢送关注:李永宁lyn

文章已收录到 github 仓库 liyongning/blog,欢送 Watch 和 Star。