亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

Vue源碼學習之初始化模塊init.js解析

2019-11-19 15:00:23
字體:
來源:轉載
供稿:網友

我們看到了VUE分了很多模塊(initMixin()stateMixin()eventsMixin()lifecycleMixin()renderMixin()),通過使用Mixin模式,都是使用了JavaScript原型繼承的原理,在Vue的原型上面增加屬性和方法。我們繼續跟著this._init(options)走,這個一點擊進去就知道了是進入了init.js文件是在initMixin函數里面給Vue原型添加的_init方法。首先來從宏觀看看這個init文件,可以看出主要是導出了兩個函數:initMixin和resolveConstructorOptions,具體作用我們一步步來討論。咋的一看這個文件,可能有些童鞋會看不明白函數參數括號里面寫的是什么鬼,這個其實是應用了flow的類型檢查,具體flow的使用這里就不介紹了,有興趣的請移步:https://flow.org/en/

我們現在來看第一個函數initMixin,Vue實例在初始化的時候就調用了這個函數,

let uid = 0export function initMixin (Vue: Class<Component>) { Vue.prototype._init = function (options?: Object) {  const vm: Component = this  // a uid  vm._uid = uid++  let startTag, endTag  /* istanbul ignore if */  【**注:istanbul 是代碼覆蓋率檢測工具,此注釋為代碼測試用**】  if (process.env.NODE_ENV !== '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 = mergeOptions(    resolveConstructorOptions(vm.constructor),    options || {},    vm   )  }  /* istanbul ignore else */  if (process.env.NODE_ENV !== 'production') {   initProxy(vm)  } else {   vm._renderProxy = vm  }  // expose real self  vm._self = vm  initLifecycle(vm)  initEvents(vm)  initRender(vm)  callHook(vm, 'beforeCreate')  initInjections(vm) // resolve injections before data/props  initState(vm)  initProvide(vm) // resolve provide after data/props  callHook(vm, 'created')  /* istanbul ignore if */  if (process.env.NODE_ENV !== '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)  } }}

我們本著宏觀簡化原則,這個函數里面前面有三個if判斷工作我們可以先不細化討論,大致第一個是用performance做性能監測,第二個合并option,第三個是做代理攔截,是ES6新特性,可參考阮一峰大神關于proxy的介紹【http://es6.ruanyifeng.com/#docs/proxy】。那么就進入了初始化函數主要點:

initLifecycle(vm) //生命周期變量初始化initEvents(vm) //事件監聽初始化initRender(vm) //初始化渲染callHook(vm, 'beforeCreate')  //回調鉤子beforeCreateinitInjections(vm) //初始化注入initState(vm)  // prop/data/computed/method/watch狀態初始化initProvide(vm)   // resolve provide after data/propscallHook(vm, 'created')   //回調鉤子created/* istanbul ignore if */if (process.env.NODE_ENV !== '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)}

這里來一個插曲start

V2.1.8及以前的版本】這里比較方便理解在生命周期created之后再做render,那么在created之前就無法獲取DOM。這也是在有些源碼解析文章里面很容易見到的分析,也是正確的

initLifecycle(vm)initEvents(vm)callHook(vm, 'beforeCreate')initState(vm)callHook(vm, 'created')initRender(vm) 

v2.1.9及以后的版本】但到這里一開始就懵逼了很久render提到beforeCreate之前去了,那豈不是DOM在beforeCreate之前就能獲取到了?顯然不對了,請注意render雖然提前了,但是后面多了一個if這個if里面才獲取DOM的關鍵,這個if在2.1.8版本之前是在render函數里面的,在2.1.9之后被提出來,然后render函數提前了,至于為何提前暫未了解,此處只是踩了一個看其他源碼解析不同版本帶來的坑!

initLifecycle(vm)initEvents(vm)initRender(vm)callHook(vm, 'beforeCreate')initState(vm)callHook(vm, 'created')if (vm.$options.el) { vm.$mount(vm.$options.el)}

插曲end,繼續

1.initLifecycle

function initLifecycle (vm: Component) { const options = vm.$options // locate first non-abstract parent let parent = options.parent  //我理解為父實例或者父組件 if (parent && !options.abstract) {  //例子中沒有parent,斷點代碼的時候自動跳過  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.$parent和vm.$children這倆個實例屬性,我此處沒有就跳過,其他的就是新增了一些實例屬性

2.initEvents

function initEvents (vm: Component) { vm._events = Object.create(null) vm._hasHookEvent = false // init parent attached events const listeners = vm.$options._parentListeners if (listeners) {  updateComponentListeners(vm, listeners) }}

又新增兩個屬性,后面那個if條件里面是有父組件的事件時初始化,估計就是props和events父子組件通信的事件內容。

3.initRender

function initRender (vm: Component) { vm._vnode = null // the root of the child tree vm._staticTrees = null const parentVnode = vm.$vnode = vm.$options._parentVnode const renderContext = parentVnode && parentVnode.context vm.$slots = resolveSlots(vm.$options._renderChildren, renderContext) vm.$scopedSlots = emptyObject  vm._c = (a, b, c, d) => createElement(vm, a, b, c, d, false)   vm.$createElement = (a, b, c, d) => createElement(vm, a, b, c, d, true) const parentData = parentVnode && parentVnode.data   /* istanbul ignore else */ if (process.env.NODE_ENV !== 'production') {  defineReactive(vm, '$attrs', parentData && parentData.attrs, () => {   !isUpdatingChildComponent && warn(`$attrs is readonly.`, vm)  }, true)  defineReactive(vm, '$listeners', vm.$options._parentListeners, () => {   !isUpdatingChildComponent && warn(`$listeners is readonly.`, vm)  }, true) } else {  defineReactive(vm, '$attrs', parentData && parentData.attrs, null, true)  defineReactive(vm, '$listeners', vm.$options._parentListeners, null, true) }}

此函數也是初始化了節點屬性信息,綁定createElement函數到實例【并未掛載】,接下來調用beforeCreate回調鉤子;――TODO1:后續專題分析VUE渲染邏輯

4.initInjections

function initInjections (vm: Component) { const result = resolveInject(vm.$options.inject, vm) 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])   }  })  observerState.shouldConvert = true }}

此函數也是當有inject屬性時做處理,源碼例子無inject斷點跑暫時跳過

5.initState

function initState (vm: Component) { vm._watchers = [] const opts = vm.$options if (opts.props) initProps(vm, opts.props) if (opts.methods) initMethods(vm, opts.methods) if (opts.data) {  initData(vm) } else {  observe(vm._data = {}, true /* asRootData */) } if (opts.computed) initComputed(vm, opts.computed) if (opts.watch && opts.watch !== nativeWatch) {  initWatch(vm, opts.watch) }}

可以看出此處是對options傳入的props/methods/data/computed/watch屬性做初始化――――TODO2:分析每個屬性的初始化

6.initProvide

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

這個函數跟4.initInjections在同一個inject.js中,也是在傳入參數有provide屬性時做處理,暫時跳過,然后就到了created回調鉤子,最后的vm.$mount接入TODO1;

今天initMixin到此結束,以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持武林網。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲无线码在线一区观看| 欧美美最猛性xxxxxx| 国产剧情久久久久久| 91网站在线看| 欧美亚洲国产日本| 精品一区二区三区电影| 欧美极品美女视频网站在线观看免费| 中文字幕日韩欧美在线| 国产成人精品久久二区二区91| 亚洲欧美制服综合另类| 日韩在线播放av| 精品成人69xx.xyz| 欧美一级淫片videoshd| 亚洲aⅴ男人的天堂在线观看| 欧美伊久线香蕉线新在线| 亚洲亚裔videos黑人hd| www.日本久久久久com.| 上原亚衣av一区二区三区| 欧美中文字幕在线播放| 亚洲专区国产精品| 国产精品91久久久久久| 色哟哟亚洲精品一区二区| 国产中文字幕日韩| 国产91久久婷婷一区二区| 日韩国产高清视频在线| 97视频网站入口| 亚洲精品动漫久久久久| 久久久久久久香蕉网| 国产一区二区美女视频| 久久亚洲一区二区三区四区五区高| 伊人伊成久久人综合网站| 亚洲**2019国产| 国产精品自产拍在线观看| 国内精品伊人久久| 岛国视频午夜一区免费在线观看| 日本成人免费在线| 色婷婷综合成人av| 精品偷拍一区二区三区在线看| 亚洲老司机av| 久久久亚洲福利精品午夜| 中文字幕久热精品在线视频| 中文字幕亚洲欧美日韩高清| 九九久久久久久久久激情| 国产精品久久久久久久久久久久久| 综合国产在线视频| 国产成人精品在线观看| 国产精品一区专区欧美日韩| 欧美精品激情blacked18| 亚洲精品乱码久久久久久按摩观| 欧美视频在线观看免费网址| 日av在线播放中文不卡| xxxxxxxxx欧美| 日韩精品极品视频免费观看| 久久久久www| 色婷婷av一区二区三区久久| 久久精品中文字幕电影| 国产精品久久97| 成人欧美一区二区三区在线湿哒哒| 欧美亚洲免费电影| 亚洲国产91精品在线观看| 欧美激情二区三区| 成人动漫网站在线观看| 亚洲影院高清在线| 欧美孕妇与黑人孕交| 亚洲第一在线视频| 国产综合视频在线观看| 久久久91精品| 韩剧1988免费观看全集| 亚洲美女av在线| 92国产精品视频| 精品国产一区二区三区四区在线观看| 91免费精品国偷自产在线| 亚洲欧美激情四射在线日| 欧美日韩一区免费| 欧美在线视频a| 欧美成aaa人片在线观看蜜臀| 欧美日韩在线视频首页| 国产精品久久久久99| 成人黄色免费看| 国内精品小视频在线观看| 欧美精品在线网站| 亚洲精品av在线| 欧美与黑人午夜性猛交久久久| 欧美性猛交xxxx乱大交蜜桃| 欧美午夜精品久久久久久浪潮| 亚洲3p在线观看| 91在线色戒在线| 亚洲国产精品字幕| 国内精久久久久久久久久人| 国产精品爽爽爽| 国产欧美久久久久久| 午夜伦理精品一区| 亚洲国产成人久久综合| 97在线视频国产| 亚洲欧洲在线观看| 国产精品揄拍一区二区| 人人澡人人澡人人看欧美| 国产精品高潮呻吟久久av黑人| 欧美在线视频a| 亚洲欧美综合区自拍另类| 欧美亚洲另类视频| 日韩国产欧美精品在线| 欧美精品激情在线观看| 日韩欧美精品网址| 久久九九亚洲综合| 久久久亚洲福利精品午夜| 2019中文字幕全在线观看| 欧美日韩成人在线视频| 2019亚洲日韩新视频| 91精品国产高清| 26uuu日韩精品一区二区| 欧美日韩一区免费| 久久99久久99精品免观看粉嫩| 欧美大秀在线观看| 高清日韩电视剧大全免费播放在线观看| 国产精品综合不卡av| 2019亚洲男人天堂| 国产精品美女网站| 欧美乱妇高清无乱码| 91极品女神在线| 国产日韩精品在线播放| 97涩涩爰在线观看亚洲| 国内精品一区二区三区| 97人洗澡人人免费公开视频碰碰碰| 欧美最顶级丰满的aⅴ艳星| 日韩成人中文电影| 欧美日韩国产影院| 欧美性猛交xxxx偷拍洗澡| xxx成人少妇69| 亚洲免费电影在线观看| 欧美精品免费在线| 国产精品久久久久不卡| 国产视频久久久久久久| 亚洲视频在线看| 亚洲香蕉在线观看| 超碰97人人做人人爱少妇| 黑人巨大精品欧美一区二区| 午夜精品久久久久久久久久久久久| 日韩在线免费视频| 久久久精品久久久久| 国产精品激情av电影在线观看| 久久国产精品久久精品| 欧美一级大片在线免费观看| 欧美午夜视频在线观看| 国产精品久久久久久婷婷天堂| 亚洲欧美成人网| 91久久精品一区| 久久91亚洲人成电影网站| 中文字幕日韩在线视频| 视频一区视频二区国产精品| 92国产精品久久久久首页| 欧美激情aaaa| 国产在线999| 91tv亚洲精品香蕉国产一区7ujn| 亚洲视频电影图片偷拍一区| 国产精品久久久久久久久久| 欧美成年人视频网站欧美| 国产一区二区日韩精品欧美精品| 欧美日韩一二三四五区| 亚洲成年人在线| 国产精品自拍偷拍视频| 日韩中文字幕免费视频| yellow中文字幕久久| 91在线视频导航|