• <fieldset id="8imwq"><menu id="8imwq"></menu></fieldset>
  • <bdo id="8imwq"><input id="8imwq"></input></bdo>
    最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
    問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
    當前位置: 首頁 - 科技 - 知識百科 - 正文

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

    來源:懂視網 責編:小采 時間:2020-11-27 22:26:18
    文檔

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

    Vue源碼學習之初始化模塊init.js解析:我們看到了VUE分了很多模塊(initMixin()stateMixin()eventsMixin()lifecycleMixin()renderMixin()),通過使用Mixin模式,都是使用了JavaScript原型繼承的原理,在Vue的原型上面增加屬性和方法。我們繼續跟著this._init(options)走,這
    推薦度:
    導讀Vue源碼學習之初始化模塊init.js解析:我們看到了VUE分了很多模塊(initMixin()stateMixin()eventsMixin()lifecycleMixin()renderMixin()),通過使用Mixin模式,都是使用了JavaScript原型繼承的原理,在Vue的原型上面增加屬性和方法。我們繼續跟著this._init(options)走,這

    我們看到了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 = 0
    
    export 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') //回調鉤子beforeCreate
    initInjections(vm) //初始化注入
    initState(vm) // prop/data/computed/method/watch狀態初始化
    initProvide(vm) // resolve provide after data/props
    callHook(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到此結束,

    聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

    文檔

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

    Vue源碼學習之初始化模塊init.js解析:我們看到了VUE分了很多模塊(initMixin()stateMixin()eventsMixin()lifecycleMixin()renderMixin()),通過使用Mixin模式,都是使用了JavaScript原型繼承的原理,在Vue的原型上面增加屬性和方法。我們繼續跟著this._init(options)走,這
    推薦度:
    標簽: VUE js 模塊
    • 熱門焦點

    最新推薦

    猜你喜歡

    熱門推薦

    專題
    Top
    主站蜘蛛池模板: 国产乱子伦精品无码码专区| 91精品全国免费观看青青| 日产精品久久久一区二区| 国产精品第一页在线| 久久er99热精品一区二区| 国内精品久久久久久久影视麻豆| 精品久久久久久久无码| 亚洲精品成人区在线观看| 国产999精品久久久久久| 91精品在线播放| 久久亚洲中文字幕精品有坂深雪 | 亚洲av无码国产精品夜色午夜| 国产一区精品| 欧美巨大黑人精品videos| 国产成人亚洲综合无码精品| 亚洲av无码乱码国产精品fc2| 欧美在线精品一区二区三区 | 国产精品福利在线观看免费不卡 | 国产精品一区在线播放| 久久99精品久久久久久久不卡| 亚洲国产精品乱码一区二区 | 国产精品热久久毛片| 91po国产在线精品免费观看| 精品国产污污免费网站| 国产99视频精品一区| 国产精品无码AV一区二区三区| 久久精品aⅴ无码中文字字幕重口| 亚洲国产精品无码久久SM| 亚洲精品国产成人片| 真实国产乱子伦精品视频| 日本熟妇亚洲欧美精品区| 久久久久久无码国产精品中文字幕| 国产精品麻豆VA在线播放| 国产韩国精品一区二区三区| 丰满人妻熟妇乱又伦精品劲| 99精品国产成人一区二区| 亚洲精品无码不卡| 99久久婷婷国产综合精品草原 | 国产精品女同久久久久电影院| 国内精品伊人久久久久av一坑| 成人精品一区二区三区中文字幕|