vue源码阅读——vue生命周期

这是官网上一个最简单的例子

<div id="app">
  {{ message }}
</div>

<script>
    var app = new Vue({
      el: '#app',
      data: {
        message: 'Hello Vue!'
      }
   })
 </script>

接下来我通过在chrome中一步步运行代码来理清其内部逻辑。
首先当然是调用构造函数

function Vue (options) {
  if (process.env.NODE_ENV !== 'production' &&
    !(this instanceof Vue)
  ) {
    warn('Vue is a constructor and should be called with the `new` keyword')
  }
  this._init(options) // 主要就是这个步骤,调用原型中的_init方法
}

//下面的这些方法都是给Vue.prototype增加方法和属性的,具体意义看vue官网api就能了解
initMixin(Vue) // 给原型增加了_init()
stateMixin(Vue) // 增加$data,$props,$set(),$delete(),$watch()
eventsMixin(Vue) // 增加$on(),$once(),$off(),$emit()
lifecycleMixin(Vue) // 增加_update(),$forceUpdate(),$destroy()
renderMixin(Vue) // 增加_render()

然后就进入_init()方法了

var uid$1 = 0;

function initMixin (Vue) {
  Vue.prototype._init = function (options) {
    var vm = this; // // 定义一个vm指向Vue实例(这里就是app)
    // a uid
    vm._uid = uid$1++;

    var startTag, endTag;
    /* istanbul ignore if */
    if ("development" !== 'production' && config.performance && mark) {
      startTag = "vue-perf-init:" + (vm._uid);
      endTag = "vue-perf-end:" + (vm._uid);
      mark(startTag);
    }

    // a flag to avoid this being observed
    vm._isVue = true;
    // merge options
    if (options && options._isComponent) { // 这个例子不是组件,所以先跳过
      // optimize internal component instantiation
      // since dynamic options merging is pretty slow, and none of the
      // internal component options needs special treatment.
      initInternalComponent(vm, options);
    } else {
      // 这个步骤炒鸡重要,给vm增加了一个$options属性,这个属性是合并之后的options。详细内容看¹mergeOptions
      vm.$options = mergeOptions(
        resolveConstructorOptions(vm.constructor),
        options || {},
        vm
      );
    }
    /* istanbul ignore else */
    {
      initProxy(vm);
    }
    // expose real self
    vm._self = vm;
    // 下面这些都是和vue生命周期有关的一些操作了
    initLifecycle(vm) // ²初始化生命周期
    initEvents(vm) // ³初始化事件
    initRender(vm) // ⁴初始化render
    callHook(vm, 'beforeCreate') // ⁵beforeCreate
    initInjections(vm) // ⁶初始化inject
    initState(vm) // ⁷对data/props/computed/watch等进行监听
    initProvide(vm) // ⁸初始化provide
    callHook(vm, 'created') // ⁹create

    /* istanbul ignore if */
    if ("development" !== 'production' && config.performance && mark) {
      vm._name = formatComponentName(vm, false);
      mark(endTag);
      measure(((vm._name) + " init"), startTag, endTag);
    }

    if (vm.$options.el) {
      vm.$mount(vm.$options.el); // ¹⁰挂载
    }
  };
}

1.mergeOptions
这个方法顾名思义就是合并分支的。那我们先看没合并之前的vm

mergeOptions执行前

再看这个方法运行之后变成啥样了


mergeOptions执行后

明显能注意到的就是,vm.$options比之前的options多了components,directives,filters,_base,然后data变成一个方法了。因此我们大概知道这个方法就是添加vm.$options然后把基础的options和传入的options合并(其实看函数名就差不多知道了)

2.initLifecycle

initLifecycle执行前
initLifecycle执行后

对比很明显,增加了很多属性,来看一眼代码

function initLifecycle (vm: Component) {
  const options = vm.$options

  // locate first non-abstract parent
  let parent = options.parent
  if (parent && !options.abstract) {
    while (parent.$options.abstract && parent.$parent) {
      parent = parent.$parent
    }
    parent.$children.push(vm)
  }

  vm.$parent = parent
  vm.$root = parent ? parent.$root : vm // 没有父实例的话把根实例设成自己了

  vm.$children = []
  vm.$refs = {}

  vm._watcher = null
  vm._inactive = null
  vm._directInactive = false
  vm._isMounted = false
  vm._isDestroyed = false
  vm._isBeingDestroyed = false
}

看样子这个函数是很好理解的,主要目的是为了给vm增加$children、$parent、$refs、$root。然后以下划线开始的这些属性看命名就能才出来是与vue生命周期有关的一些变量(下面我把以$开头的称为实例属性,把_开头的称变量)。这里放一张官网上的生命周期图片,具体的在之后的内容中也会涉及到的

vue生命周期
  1. initEvents


    initEvents执行前
initEvents执行后

看上去也就是多了几个变量,我们看看具体的函数执行情况吧

initEvents代码.png

这么一张图看上去已经很清晰了,vm.$options._parentListeners未定义,因此之后的步骤也不会执行的,整个函数就新增了两个变量。不过这个函数本来想干的事,看作者注释的话是想添加父实例的事件,这个功能在我之后调试到相应例子之后会再进行扩散的

4.initRender

initRender执行前

initRender执行后

好害怕啊突然变长好多~总结一下就是又多了很多属性($attrs$listeners、$slots、$scopedSlots、$vnode)和变量,还增加了方法($createElement)。先看一下执行过程

initRender代码

这里大致可以分为三个部分,第一又增加了变量和属性,同样我们在这个例子里看不出什么,比如说$slots这个实例属性,第二是给vm增加了一个$createElement方法,调用这个方法其实就是调用createElement方法,看命名就知道是一个生成元素的方法。第三个是调用了一个defineReactive方法,这个方法是什么意思呢

defineReactive

这些步骤基本是在做初始化

defineReactive

但是最重要的这里,事实上对于get,在我们这个例子中,Dep.target为null,因此get就是返回了value,而set也就是普通的设置了值而已,那么这么长的代码有什么用呢,这其实又和数据绑定有关。关于数据绑定我这里也先不再延伸。所以这第三部分就是在给$attrs、$listeners重写get和set

5.callHook(vm, 'beforeCreate')

function callHook (vm, hook) {
  var handlers = vm.$options[hook];
  if (handlers) {
    for (var i = 0, j = handlers.length; i < j; i++) {
      try {
        handlers[i].call(vm); // 执行beforeCreate下的每个函数
      } catch (e) {
        handleError(e, vm, (hook + " hook"));
      }
    }
  }
  if (vm._hasHookEvent) {
    vm.$emit('hook:' + hook);
  }
}

这个函数很简单,我们这个简单的vue实例也没有传beforeCreate这个选项,因此这步也可以忽略,只需要知道在create之前会调用这样一个东西

6.initInjections

 function initInjections (vm: Component) {
  const result = resolveInject(vm.$options.inject, vm) //获得vm.$options.inject的所有属性
  if (result) {
    observerState.shouldConvert = 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]) // 把inject监听起来
      }
    })
    observerState.shouldConvert = true
  }
}

瞅瞅这个简单的代码,对于我们这个例子来说,又是可以跳过的。不过我们还需大致看看这个函数,它其实是对inject选项的初始化,这个眼熟的defineReactive,没错就是对inject的每个属性的get,set都重写了一下。

7.initState

initState执行前

这个函数明显是很重要的,因为它处理了props/methods/data/computed/watch等选项,而这些选项又是我们经常需要用到的。那我们从源码看看它怎么处理的

function initState (vm: Component) {
  // 首先在vm上初始化一个_watchers数组,缓存这个vm上的所有watcher
  vm._watchers = []
  // 获取options,包括在new Vue传入的,同时还包括了Vue所继承的options
  const opts = vm.$options
  // 初始化props属性
  if (opts.props) initProps(vm, opts.props)
  // 初始化methods属性
  if (opts.methods) initMethods(vm, opts.methods)
  // 初始化data属性
  if (opts.data) {
    initData(vm)
  } else {
    observe(vm._data = {}, true /* asRootData */)
  }
  // 初始化computed属性
  if (opts.computed) initComputed(vm, opts.computed)
  // 初始化watch属性
  if (opts.watch && opts.watch !== nativeWatch) {
    initWatch(vm, opts.watch)
  }
}

这段代码从命名上看是很简单的,没有阅读障碍。那我们看看和我们相关的initData

function initData (vm: Component) {
  // 获取data
  let data = vm.$options.data
  // 因为data有可能是个函数,这里给它转换了一下
  data = vm._data = typeof data === 'function'
    ? getData(data, vm)
    : data || {}
  if (!isPlainObject(data)) {
    data = {}
    process.env.NODE_ENV !== 'production' && warn(
      'data functions should return an object:\n' +
      'https://vuejs.org/v2/guide/components.html#data-Must-Be-a-Function',
      vm
    )
  }
  // proxy data on instance
  // 收集data的自身属性名
  const keys = Object.keys(data)
  const props = vm.$options.props
  const methods = vm.$options.methods
  // 遍历每个属性
  let i = keys.length
  while (i--) {
    const key = keys[i]
    if (process.env.NODE_ENV !== 'production') {
      // data的属性不能同时在method中
      if (methods && hasOwn(methods, key)) {
        warn(
          `method "${key}" has already been defined as a data property.`,
          vm
        )
      }
    }
    // data的属性不能同时在props中
    if (props && hasOwn(props, key)) {
      process.env.NODE_ENV !== 'production' && warn(
        `The data property "${key}" is already declared as a prop. ` +
        `Use prop default value instead.`,
        vm
      )
    } else if (!isReserved(key)) {
      // 给data里的每个属性重写了get和set
      proxy(vm, `_data`, key)
    }
  }
  // 时刻观察data
  // observe data
  observe(data, true /* asRootData */)
}

这个observe函数实际上做了很多事情。
8.initProvide

function initProvide (vm: Component) {
  const provide = vm.$options.provide
  if (provide) {
    vm._provided = typeof provide === 'function'
      ? provide.call(vm)
      : provide
  }
}

初始化provide选项。超简单的代码,谁都能看懂,加了个_provided变量,和我这个例子好像也没有什么关系

  1. callHook(vm, 'created')
    和callHook(vm, 'beforeCreate')一样一样的,不再赘述了

10.mount

if (vm.$options.el) {
      vm.$mount(vm.$options.el) //如果传入了el就挂载
    }

像这段代码,和生命周期图结合起来看,就一目了然,如果有el选项的话就挂载

$mount-1
$mount-2
mount
mountComponent核心代码
Wacher-1
Wacher-2
get

那么再来看看updateComponent


updateComponent

这里首先要执行vm._render()

_render-1

回到_update

_update

我们传了旧的和新的node进去


__patch__

这步执行完页面上是这样的

两个

直到下图执行完后,页面才算更新完毕

去掉一个
好了

说的可能还有点不太完整。整个_init方法实际上只是到mount部分为止。真的要销毁的话得手动调用$destroy。还有很多函数没有仔细扩散,等之后调试到相关例子的时候会再补充。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 一、Native 每一个页面都是一个 instance,framework 是全局唯一的,js 和 native ...
    Yang152412阅读 5,647评论 0 50
  • 前言 人生苦多,快来 Kotlin ,快速学习Kotlin! 什么是Kotlin? Kotlin 是种静态类型编程...
    任半生嚣狂阅读 26,179评论 9 118
  • 十月
    阿洛特斯阅读 153评论 0 3
  • 一、前言 是否也有一颗焦虑的心:想学点什么却不知道学什么好?是否也有一颗迷茫的心:在学习的道路上举步维艰,蹒跚蹉跎...
    夜_阑珊阅读 2,724评论 2 18