一文了解Vue實例掛載的過程
new Vue()這個過程中究竟做了些什么?
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)
}vue構建函數調用_init方法,但發(fā)現本文件中并沒有此方法,但仔細可以看到文件下方定義了很多初始化方法
initMixin(Vue); // 定義 _init stateMixin(Vue); // 定義 $set $get $delete $watch 等 eventsMixin(Vue); // 定義事件 $on $once $off $emit lifecycleMixin(Vue);// 定義 _update $forceUpdate $destroy renderMixin(Vue); // 定義 _render 返回虛擬dom
首先可以看initMixin方法,發(fā)現該方法在Vue原型上定義了_init方法
Vue.prototype._init = function (options?: Object) {
const vm: Component = this
vm._uid = uid++
let startTag, endTag
if (process.env.NODE_ENV !== 'production' && config.performance && mark) {
startTag = `vue-perf-start:${vm._uid}`
endTag = `vue-perf-end:${vm._uid}`
mark(startTag)
}
vm._isVue = true
// 合并屬性,判斷初始化的是否是組件,這里合并主要是 mixins 或 extends 的方法
if (options && options._isComponent) {
initInternalComponent(vm, options)
} else { // 合并vue屬性
vm.$options = mergeOptions(
resolveConstructorOptions(vm.constructor),
options || {},
vm
)
}
/* istanbul ignore else */
if (process.env.NODE_ENV !== 'production') {
// 初始化proxy攔截器
initProxy(vm)
} else {
vm._renderProxy = vm
}
// expose real self
vm._self = vm
// 初始化組件生命周期標志位
initLifecycle(vm)
// 初始化組件事件偵聽
initEvents(vm)
// 初始化渲染方法
initRender(vm)
callHook(vm, 'beforeCreate')
// 初始化依賴注入內容,在初始化data、props之前
initInjections(vm) // resolve injections before data/props
// 初始化props/data/method/watch/methods
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(`vue ${vm._name} init`, startTag, endTag)
}
// 掛載元素
if (vm.$options.el) {
vm.$mount(vm.$options.el)
}
}在 init() 中 會通過 callHook(vm,beforeCreate) 來執(zhí)行 beforeCreate 生命周期函數然后通過initState(vm) 初始化 props 、 methods 、data 接著會通過callHook(vm, 'created') 來執(zhí)行 created 生命周期函數 最后通過 vm.$mount(vm.$options.el) 來掛載元素
所以:
1. 在beforeCreate 生命周期函數中是無法訪問 props和data 因為他們還沒有被初始化
2. 同理在created函數中可以訪問props,methods、data數據同時也是最早可以調用接口的生
命周期函數,但是此時dom并未掛載 所以無法訪問dom元素
3. 在mounted中 此時dom已經掛載成功 所以可以訪問dom元素也是最早可以操作dom元素的生命周期函數
初始化數據 initState(vm)
export function initState (vm: Component) {
// 初始化組件的watcher列表
vm._watchers = []
const opts = vm.$options
// 初始化props
if (opts.props) initProps(vm, opts.props)
// 初始化methods方法
if (opts.methods) initMethods(vm, opts.methods)
if (opts.data) {
// 初始化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)
}
}在initState中會先通過initProps 、 initMethods 、 initData 先后分別來初始化 相關數據 在這里會 初始化組件的watcher列表
看下 initData
function initData (vm: Component) {
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
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') {
// 屬性名不能與方法名重復
if (methods && hasOwn(methods, key)) {
warn(
`Method "${key}" has already been defined as a data property.`,
vm
)
}
}
// 屬性名不能與state名稱重復
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)) { // 驗證key值的合法性
// 將_data中的數據掛載到組件vm上,這樣就可以通過this.xxx訪問到組件上的數據
proxy(vm, `_data`, key)
}
}
// observe data
// 響應式監(jiān)聽data是數據的變化
observe(data, true /* asRootData */)
}需要注意的是:初始化data數據的時候 會校驗 props中變量名稱和data中的不能重復,在這里會通過observe劫持data的所有屬性,如果監(jiān)聽到數據變化就通知訂閱者watcher來更新數據,以此來實現數據雙向綁定 ,這就是vue實現數據響應式的 發(fā)布訂閱者模式 挖個坑自己實現該模式
再看下掛載方法是調用vm.$mount
Vue.prototype.$mount = function (
el?: string | Element,
hydrating?: boolean
): Component {
// 獲取或查詢元素
el = el && query(el)
/* istanbul ignore if */
// vue 不允許直接掛載到body或頁面文檔上
if (el === document.body || el === document.documentElement) {
process.env.NODE_ENV !== 'production' && warn(
`Do not mount Vue to <html> or <body> - mount to normal elements instead.`
)
return this
}
const options = this.$options
// resolve template/el and convert to render function
if (!options.render) {
let template = options.template
// 存在template模板,解析vue模板文件
if (template) {
if (typeof template === 'string') {
if (template.charAt(0) === '#') {
template = idToTemplate(template)
/* istanbul ignore if */
if (process.env.NODE_ENV !== 'production' && !template) {
warn(
`Template element not found or is empty: ${options.template}`,
this
)
}
}
} else if (template.nodeType) {
template = template.innerHTML
} else {
if (process.env.NODE_ENV !== 'production') {
warn('invalid template option:' + template, this)
}
return this
}
} else if (el) {
// 通過選擇器獲取元素內容
template = getOuterHTML(el)
}
if (template) {
/* istanbul ignore if */
if (process.env.NODE_ENV !== 'production' && config.performance && mark) {
mark('compile')
}
/**
* 1.將temmplate解析ast tree
* 2.將ast tree轉換成render語法字符串
* 3.生成render方法
*/
const { render, staticRenderFns } = compileToFunctions(template, {
outputSourceRange: process.env.NODE_ENV !== 'production',
shouldDecodeNewlines,
shouldDecodeNewlinesForHref,
delimiters: options.delimiters,
comments: options.comments
}, this)
options.render = render
options.staticRenderFns = staticRenderFns
/* istanbul ignore if */
if (process.env.NODE_ENV !== 'production' && config.performance && mark) {
mark('compile end')
measure(`vue ${this._name} compile`, 'compile', 'compile end')
}
}
}
return mount.call(this, el, hydrating)
}在調用vm.$mount方法時 會將template 解析為 抽象語法樹 (ast tree) 再將抽象語法樹 轉換成render語法字符串 最終生成render方法 掛載到vm上后,會再次調用mount方法
Vue.prototype.$mount = function (
el?: string | Element,
hydrating?: boolean
): Component {
el = el && inBrowser ? query(el) : undefined
// 渲染組件
return mountComponent(this, el, hydrating)
}調用mountComponent渲染組件
export function mountComponent (
vm: Component,
el: ?Element,
hydrating?: boolean
): Component {
vm.$el = el
// 如果沒有獲取解析的render函數,則會拋出警告
// render是解析模板文件生成的
if (!vm.$options.render) {
vm.$options.render = createEmptyVNode
if (process.env.NODE_ENV !== 'production') {
/* istanbul ignore if */
if ((vm.$options.template && vm.$options.template.charAt(0) !== '#') ||
vm.$options.el || el) {
warn(
'You are using the runtime-only build of Vue where the template ' +
'compiler is not available. Either pre-compile the templates into ' +
'render functions, or use the compiler-included build.',
vm
)
} else {
// 沒有獲取到vue的模板文件
warn(
'Failed to mount component: template or render function not defined.',
vm
)
}
}
}
// 執(zhí)行beforeMount鉤子
callHook(vm, 'beforeMount')
let updateComponent
/* istanbul ignore if */
if (process.env.NODE_ENV !== 'production' && config.performance && mark) {
updateComponent = () => {
const name = vm._name
const id = vm._uid
const startTag = `vue-perf-start:${id}`
const endTag = `vue-perf-end:${id}`
mark(startTag)
const vnode = vm._render()
mark(endTag)
measure(`vue ${name} render`, startTag, endTag)
mark(startTag)
vm._update(vnode, hydrating)
mark(endTag)
measure(`vue ${name} patch`, startTag, endTag)
}
} else {
// 定義更新函數
updateComponent = () => {
// 實際調?是在lifeCycleMixin中定義的_update和renderMixin中定義的_render
vm._update(vm._render(), hydrating)
}
}
// 監(jiān)聽當前組件狀態(tài),當有數據變化時,更新組件
new Watcher(vm, updateComponent, noop, {
before () {
if (vm._isMounted && !vm._isDestroyed) {
// 數據更新引發(fā)的組件更新
callHook(vm, 'beforeUpdate')
}
}
}, true /* isRenderWatcher */)
hydrating = false
if (vm.$vnode == null) {
vm._isMounted = true
callHook(vm, 'mounted')
}
return vm
}此時會觸發(fā)beforeMount鉤子函數 ,定義updateComponent來渲染頁面視圖的方法,監(jiān)聽組件數據,一旦發(fā)生變化,觸發(fā)beforeUpdate生命鉤子 最后執(zhí)行 callHook(vm, 'mounted') 鉤子函數 完成掛載 updateComponent方法主要執(zhí)行在vue初始化時聲明的render,update方法
render的作用主要是生成vnode
// 定義vue 原型上的render方法
Vue.prototype._render = function (): VNode {
const vm: Component = this
// render函數來自于組件的option
const { render, _parentVnode } = vm.$options
if (_parentVnode) {
vm.$scopedSlots = normalizeScopedSlots(
_parentVnode.data.scopedSlots,
vm.$slots,
vm.$scopedSlots
)
}
vm.$vnode = _parentVnode
// render self
let vnode
try {
currentRenderingInstance = vm
// 調用render方法,自己的獨特的render方法, 傳入createElement參數,生成vNode
vnode = render.call(vm._renderProxy, vm.$createElement)
} catch (e) {
handleError(e, vm, `render`)
if (process.env.NODE_ENV !== 'production' && vm.$options.renderError) {
try {
vnode = vm.$options.renderError.call(vm._renderProxy, vm.$createElement, e)
} catch (e) {
handleError(e, vm, `renderError`)
vnode = vm._vnode
}
} else {
vnode = vm._vnode
}
} finally {
currentRenderingInstance = null
}
// if the returned array contains only a single node, allow it
if (Array.isArray(vnode) && vnode.length === 1) {
vnode = vnode[0]
}
// return empty vnode in case the render function errored out
if (!(vnode instanceof VNode)) {
if (process.env.NODE_ENV !== 'production' && Array.isArray(vnode)) {
warn(
'Multiple root nodes returned from render function. Render function ' +
'should return a single root node.',
vm
)
}
vnode = createEmptyVNode()
}
// set parent
vnode.parent = _parentVnode
return vnode
}_update主要功能是調用patch,將vnode轉換為真實DOM,并且更新到頁面中
Vue.prototype._update = function (vnode: VNode, hydrating?: boolean) {
const vm: Component = this
const prevEl = vm.$el
const prevVnode = vm._vnode
// 設置當前激活的作用域
const restoreActiveInstance = setActiveInstance(vm)
vm._vnode = vnode
// Vue.prototype.__patch__ is injected in entry points
// based on the rendering backend used.
if (!prevVnode) {
// initial render
// 執(zhí)行具體的掛載邏輯
vm.$el = vm.__patch__(vm.$el, vnode, hydrating, false /* removeOnly */)
} else {
// updates
vm.$el = vm.__patch__(prevVnode, vnode)
}
restoreActiveInstance()
// update __vue__ reference
if (prevEl) {
prevEl.__vue__ = null
}
if (vm.$el) {
vm.$el.__vue__ = vm
}
// if parent is an HOC, update its $el as well
if (vm.$vnode && vm.$parent && vm.$vnode === vm.$parent._vnode) {
vm.$parent.$el = vm.$el
}
// updated hook is called by the scheduler to ensure that children are
// updated in a parent's updated hook.
}總結
到此這篇關于Vue實例掛載過程的文章就介紹到這了,更多相關Vue實例掛載過程內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
element-ui table行點擊獲取行索引(index)并利用索引更換行順序
這篇文章主要介紹了element-ui table行點擊獲取行索引(index)并利用索引更換行順序,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-02-02
vue中axios處理http發(fā)送請求的示例(Post和get)
本篇文章主要介紹了vue中axios處理http請求的示例(Post和get),這里整理了詳細的代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-10-10

