摘要:從開始看運行流程本篇文章不會具體分析很多每個方法內部具體邏輯,只為了研究一下瀏覽器加載文件后以及我后,都調用了哪些方法,這些方法都是做什么的等等。
從 new Vue()開始看vue運行流程
本篇文章不會具體分析很多每個方法內部具體邏輯,只為了研究一下瀏覽器加載vuejs文件后以及我new Vue后,都調用了哪些方法,這些方法都是做什么的等等。以便對vue的執(zhí)行流程有個大致了解,方便遇見問題排查是哪個過程出了問題。
用vue這么久我是很好奇從vue是怎么一步步把單文件組件的內容給渲染到空的div下面的,那從main.js入口開始,為了方便理解,我們只保留一個App.vue, main.js文件,路由,vuex,插件等等都不引入,在App.vue里面我們只定義一個data字段,插值綁定一個。
vue版本:2.5.2
先看下main.js,比較簡單
import Vue from "vue"; import App from "./App"; Vue.config.productionTip = false; /* eslint-disable no-new */ new Vue({ el: "#app", render: h => h(App) });
下面開始分析具體調用了哪些方法
分析過程會忽略一些輔助方法等無關的方法調用...
在new Vue之前也就是會做一些初始化的工作,列舉部分重要的代碼
/* initMixin給Vue.prototype添加: _init函數(shù), ... */ initMixin(Vue); /* stateMixin給Vue.prototype添加: $data屬性, $props屬性, $set函數(shù), $delete函數(shù), $watch函數(shù), ... */ stateMixin(Vue); /* eventsMixin給Vue.prototype添加: $on函數(shù), $once函數(shù), $off函數(shù), $emit函數(shù), $watch方法, ... */ eventsMixin(Vue); /* lifecycleMixin給Vue.prototype添加: _update方法:私有方法,用于更新dom,其中調用_patch產(chǎn)生跟新后的dom, $forceUpdate函數(shù), $destroy函數(shù), ... */ lifecycleMixin(Vue); /* renderMixin給Vue.prototype添加: $nextTick函數(shù), _render函數(shù), ... */ renderMixin(Vue);
至此,部分常見的我們經(jīng)??吹降某跏蓟ぷ饕呀?jīng)做完,現(xiàn)在我們知道我們在vm上用的$開頭的方法都是在一加載vue.js完成后就掛在了Vue.prototype,我們的vm實例就可以用這些方法了。
下面當我們正式開始new Vue()
//調用this._init() new Vue(options); /* this._init函數(shù) 依次調用了跟vm相關的初始化函數(shù): initLifecycle:給vm掛在一下屬性 $parent:undefined, $root:vm, $children:[], $refs:{}, _isMounted:false, _isDestoryed:false, _watcher:null, ... initEvents:初始化事件. initRender:主要做了一下事情: 給vm添加_c函數(shù)和$createElement,其實是createElement別名, 給vm添加$attrs和$listeners屬性,$attrs & $listeners are exposed for easier HOC creation callHook(vm, "beforeCreate"):此時我們看到觸發(fā)了beforeCreate鉤子,此時的vm有哪些屬性應該一目了然了. initInjections(vm): resolve injections before data/props,我們可以看到在初始化inject時還沒有data和props initState:主要進行初始化: data:initData(vm), props:initProps(vm,opt.props) computed:initComputed(vm,opt.computed), methods:initMethods(vm,opt.methods), watch:initWatch(vm,opt.watch) initProvide:resolve provide,根據(jù)源碼注釋我們可以知道,在provide中可以使用props和data callHook(vm, "created"):此時生命周期created觸發(fā),我們能訪問data,prop,provide等等 下面最后一步至關重要: if (vm.$options.el) { vm.$mount(vm.$options.el); } 這里是判斷我們是否傳入了el,屬性,傳入了則調用$mount方法掛載內容到el所在節(jié)點下 */ this._init(options) //在執(zhí)行完 this._init() 后進入了最最重要的一步,掛載組件 //程序接著往下走回執(zhí)行:mountComponent /* 函數(shù) mountComponent 主要做了下面這些事情: 觸發(fā) beforeMount() 鉤子函數(shù), 聲明 updateComponent 函數(shù),里面調用了 vm_update(vm._render(),...),這兩個方法作用下面執(zhí)行到的時候說一下. new Watcher(),此時會傳入updateComponent函數(shù),并隨后執(zhí)行此函數(shù),執(zhí)行后會發(fā)生一些函數(shù)執(zhí)行,我只列舉比較重要的大流程函數(shù): Vue._render:執(zhí)行由vue-loader生成的render函數(shù)或者自己寫的render函數(shù),最終返回一個由createComponent(非createPatchFunction內部的)產(chǎn)生的vnode. createComponent(非createPatchFunction內部):創(chuàng)建組件虛擬節(jié)點,此函數(shù)返回一個vnode,表示vue組件的vnode. vm._update:接收上面的vnode參數(shù),這里面會觸發(fā)VM.__patch__函數(shù),這個函數(shù)里面最終返回的結果就是我們在html頁面寫的空的div,但是里面有了真實的內容,此時頁面可以看到內容了, 觸發(fā) mount() 鉤子函數(shù),這個mount鉤子每個組件實例會在自己的insert hook中調用 */ mountComponent()
到了這里,函數(shù)moutComponent執(zhí)行完畢并且返回了vm,
等 this._init()執(zhí)行完我們在頁面就可以看到內容了
我們可以大致總結一下從new Vue開始都大致執(zhí)行了哪些重要的方法
1.new Vue(); 2.Vue.prototype._init(); 3.Vue.prototype.$mount(); 4.mountComponent(); 5.new Watcher(); 6.Watcher.prototype.get(); 7.updateComponent(); 8.Vue.prototype._render(); 9.render(); 10.createElement() 11.Vue.prototype._update();//這里面會執(zhí)行vm.$el=vm.__patch__(),最終根vm的$el就有了真實dom值 12.Vue.prototype.__patch__();//這個應該是最重要的方法了,他返回了真實的dom節(jié)點。
在 vm.__patch__中會產(chǎn)生一個屬于App.vue這個虛擬節(jié)點的實例,然后再次調用該實例的_init()方法,然后依次執(zhí)行步驟2到12,繼而完成app組件的掛載,最終new Vue出來的vm的$el,就是所有的真實dom。
在開始研究時在瀏覽器一步步調試看執(zhí)行的過程,在谷歌的調試工具中借助這個call Stack
很方便查看運行過程都調用了哪些函數(shù),在程序運行vm.__patch__后里面涉及到了這倆函數(shù) createElm,createChilren互相調用的邏輯,然后由于在這之前又調用了幾十個函數(shù),導致調試到最后按F10電腦要卡頓好久才能進入下一步,我們看到頁面可能是一瞬間的事情,但是vue幫我們做了很多事情,很感謝vue這么優(yōu)秀的框架。
最后希望大家也去自己嘗試下調試下源碼,你會有新的發(fā)現(xiàn)!文章版權歸作者所有,未經(jīng)允許請勿轉載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉載請注明本文地址:http://systransis.cn/yun/95649.html
摘要:圖在中應用三數(shù)據(jù)渲染過程數(shù)據(jù)綁定實現(xiàn)邏輯本節(jié)正式分析從到數(shù)據(jù)渲染到頁面的過程,在中定義了一個的構造函數(shù)。一、概述 vue已是目前國內前端web端三分天下之一,也是工作中主要技術棧之一。在日常使用中知其然也好奇著所以然,因此嘗試閱讀vue源碼并進行總結。本文旨在梳理初始化頁面時data中的數(shù)據(jù)是如何渲染到頁面上的。本文將帶著這個疑問一點點追究vue的思路。總體來說vue模版渲染大致流程如圖1所...
摘要:表示虛擬節(jié)點,為什么叫虛擬節(jié)點呢,因為不是真的節(jié)點。因為是對象,不管還是瀏覽器,都可以統(tǒng)一操作,從而獲得了服務端渲染原生渲染手寫渲染函數(shù)等能力減少操作。 寫文章不容易,點個贊唄兄弟專注 Vue 源碼分享,文章分為白話版和 源碼版,白話版助于理解工作原理,源碼版助于了解內部詳情,讓我們一起學習吧研究基于 Vue版本 【2.5.17】 如果你覺得排版難看,請點擊 下面鏈接 或者 拉到 下...
答案自己谷歌或百度找。 一、來源背景 面試題是來自微博@??途W(wǎng)發(fā)布的真實大廠前端面經(jīng)題目,我一直在收集題目長期一個一個的記錄下來的,可能會有重復,但基本前端的面試大綱和需要掌握的知識都在其中了,面試題僅做學習參考,學習者閱后也要用心鉆研其中的原理,重要知識需要系統(tǒng)學習、透徹學習,形成自己的知識鏈。 二、532道前端真實大廠面試題 express和koa的對比,兩者中間件的原理,koa捕獲異常多種情...
答案自己谷歌或百度找。 一、來源背景 面試題是來自微博@??途W(wǎng)發(fā)布的真實大廠前端面經(jīng)題目,我一直在收集題目長期一個一個的記錄下來的,可能會有重復,但基本前端的面試大綱和需要掌握的知識都在其中了,面試題僅做學習參考,學習者閱后也要用心鉆研其中的原理,重要知識需要系統(tǒng)學習、透徹學習,形成自己的知識鏈。 二、532道前端真實大廠面試題 express和koa的對比,兩者中間件的原理,koa捕獲異常多種情...
閱讀 2881·2019-08-30 15:44
閱讀 1913·2019-08-29 13:59
閱讀 2852·2019-08-29 12:29
閱讀 1099·2019-08-26 13:57
閱讀 3210·2019-08-26 13:45
閱讀 3342·2019-08-26 10:28
閱讀 857·2019-08-26 10:18
閱讀 1706·2019-08-23 16:52