本篇內(nèi)容主要講解“vue使用源碼分析”,感興趣的朋友不妨來(lái)看看。本文介紹的方法操作簡(jiǎn)單快捷,實(shí)用性強(qiáng)。下面就讓小編來(lái)帶大家學(xué)習(xí)“vue使用源碼分析”吧!
創(chuàng)新互聯(lián)的客戶來(lái)自各行各業(yè),為了共同目標(biāo),我們?cè)诠ぷ魃厦芮信浜?,從?chuàng)業(yè)型小企業(yè)到企事業(yè)單位,感謝他們對(duì)我們的要求,感謝他們從不同領(lǐng)域給我們帶來(lái)的挑戰(zhàn),讓我們激情的團(tuán)隊(duì)有機(jī)會(huì)用頭腦與智慧不斷的給客戶帶來(lái)驚喜。專業(yè)領(lǐng)域包括網(wǎng)站建設(shè)、成都網(wǎng)站建設(shè)、電商網(wǎng)站開(kāi)發(fā)、微信營(yíng)銷、系統(tǒng)平臺(tái)開(kāi)發(fā)。
生命周期
1.0版本
1.哪些生命周期接口
init Created beforeCompile Compiled Ready Attatched Detached beforeDestory destoryed
2.執(zhí)行順序
1. 不具有keep-alive
進(jìn)入:
init->create->beforeCompile->complied->attatched->ready
移出:
beforeDestory->detached->destoryed;
2. 具有keep-alive
第一次的時(shí)候
進(jìn)入:
init->create->beforeCompile->complied->attatched->ready
移出:
detached;
之后的每次
進(jìn)入:
attatched
移出:
detached
鉤子函數(shù)
3.鉤子函數(shù)有哪些
data
activete
deactivate
canactivate
candeactivate
4.執(zhí)行順序
進(jìn)入:
canactivate->actiavte->date
移出:
candeactivate->deactiavte
兩者一起出現(xiàn)
5.對(duì)于一個(gè)組件A里面有子組件B,當(dāng)這個(gè)組件A進(jìn)行移入和移出操作時(shí),組件之間的生命周期喝鉤子函數(shù)的執(zhí)行順序參考如下:
例如
A.vue
備注:下面括號(hào)里的是嵌套的子組件
1. 不具有keep-alive:
移入:
1. canActivate;
2. init;
3. create;
4. beforeCompile;
5. (嵌套子組件:init,create,beforeCompile,compile);
6. compile;
7. activate;
8. data;
9. attached;
10. (子組件attached);
11. (子組件ready);
12. ready;
移出:
13. canDeactivate;
14. deactivate;
15. beforeDestroy;
16. (子組件beforeDestroy);
17. (子組件destoryed);
18. detached;
19. (子組件detached);
20. destoryed;
2. 具有keep-alive:
移入:
1. canActivate;
2. activate;
3. data;
4. attached;
5. (子組件attached);
移出:
6. canDeactivate;
7. deactivate;
8. detached;
9. (子組件detached);
6.鉤子函數(shù)activate和data的執(zhí)行順序
涉及鉤子函數(shù)異步 resolve 規(guī)則:
1.如果鉤子返回一個(gè) Promise,則鉤子何時(shí) resolve 取決于該 Promise 何時(shí) resolve。
2.如果鉤子既不返回 Promise,也沒(méi)有任何參數(shù),則該鉤子將被同步 resolve。
3.如果鉤子不返回 Promise,但是有一個(gè)參數(shù)(transition),則鉤子會(huì)等到transition.next(),transition.abort()或是transition.redirect()之一被調(diào)用才 resolve。
4.在驗(yàn)證類的鉤子,比如canActivate,canDeactivate以及全局 beforeEach 鉤子中,如果返回值是一個(gè)布爾值 (Boolean),也會(huì)使得鉤子同步 resolve。
7.根據(jù)什么可以確保界面已經(jīng)更新完成,也就是說(shuō)掛在完成
執(zhí)行生命周期attached說(shuō)明已掛載
雙向綁定與渲染機(jī)制
1.數(shù)據(jù)的監(jiān)聽(tīng)和觸發(fā)(訂閱和發(fā)布o(jì)bserver)
src目錄下observer:
1. array.js
2. dep.js;(實(shí)現(xiàn)一個(gè)發(fā)布訂閱對(duì)象)
3. index.js;(利用Object.defineProperty這個(gè)API,并為此屬性設(shè)計(jì)一個(gè)特殊的 getter/setter,然后在 setter 里觸發(fā)一個(gè)函數(shù),達(dá)到監(jiān)聽(tīng)的效果);
下面是這部分的源碼
Object.defineProperty(obj, key, { enumerable: true, configurable: true, get: function reactiveGetter () { var value = getter ? getter.call(obj) : val if (Dep.target) { dep.depend() if (childOb) { childOb.dep.depend() } if (isArray(value)) { for (var e, i = 0, l = value.length; i < l; i++) { e = value[i] e && e.__ob__ && e.__ob__.dep.depend() } } } return value }, set: function reactiveSetter (newVal) { var value = getter ? getter.call(obj) : val if (newVal === value) { return } if (setter) { setter.call(obj, newVal) } else { val = newVal } childOb = observe(newVal) dep.notify() } })
簡(jiǎn)化上面的監(jiān)聽(tīng)與觸發(fā)代碼如下:
function notidy(obj,key){ console.log(key+" has changed"); console.log(key+" now is: "+obj[key]); } function ToData(key,val){ var ob=this; Object.defineProperty(ob,key,{ enumerable:true, configurable:true, get:function(){ return val; }, set:function(newval){ if(newval==val){ return; } val=newval; notidy(this,key); } }) }
src目錄下directive.js
在directive中可以看到一系列解析出來(lái)的屬性,而directive的實(shí)例化可以在utils/lifecycle.js中看到。
下面這段代碼在Directive.prototype._bind中
var watcher = this._watcher = new Watcher( this.vm, this.expression, this._update, // callback { filters: this.filters, twoWay: this.twoWay, deep: this.deep, preProcess: preProcess, postProcess: postProcess, scope: this._scope } ) // v-model with inital inline value need to sync back to // model instead of update to DOM on init. They would // set the afterBind hook to indicate that. if (this.afterBind) { this.afterBind() } else if (this.update) { this.update(watcher.value) } Directive.prototype.set = function (value) { /* istanbul ignore else */ if (this.twoWay) { this._withLock(function () { this._watcher.set(value) }) } else if (process.env.NODE_ENV !== 'production') { warn( 'Directive.set() can only be used inside twoWay' + 'directives.' ) } }
src目錄下Watch.js:
從下面的代碼可以找到watcher對(duì)象通過(guò)addDep方法實(shí)現(xiàn)訂閱
Watcher.prototype.addDep = function (dep) { var id = dep.id if (!this.newDepIds.has(id)) { this.newDepIds.add(id) this.newDeps.push(dep) if (!this.depIds.has(id)) { dep.addSub(this) } } }
2.前面說(shuō)那么多關(guān)于雙向綁定,其實(shí)這也是VUE內(nèi)部的渲染機(jī)制,總結(jié)如下
1. 通過(guò) observer 對(duì) data 進(jìn)行了監(jiān)聽(tīng),并且提供訂閱某個(gè)數(shù)據(jù)項(xiàng)的變化的能力
2. 把 template 解析成一段 document fragment,然后解析其中的 directive,得到每一個(gè) directive 所依賴的數(shù)據(jù)項(xiàng)及其更新方法。比如 v-text="message" 被解析之后 (這里僅作示意,實(shí)際程序邏輯會(huì)更嚴(yán)謹(jǐn)而復(fù)雜):所依賴的數(shù)據(jù)項(xiàng)this.$data.message,以及相應(yīng)的視圖更新方法 node.textContent = this.$data.message
3. 通過(guò) watcher 把上述兩部分結(jié)合起來(lái),即把 directive 中的數(shù)據(jù)依賴訂閱在對(duì)應(yīng)數(shù)據(jù)的 observer 上,這樣當(dāng)數(shù)據(jù)變化的時(shí)候,就會(huì)觸發(fā) observer,進(jìn)而觸發(fā)相關(guān)依賴對(duì)應(yīng)的視圖更新方法,最后達(dá)到模板原本的關(guān)聯(lián)效果。
3.vue是如何改進(jìn)了v-for具有相同數(shù)據(jù)渲染出錯(cuò)的?
數(shù)組的渲染
未使用track-by的數(shù)組渲染內(nèi)部緩存的默認(rèn)id是數(shù)組的值value,意味著如果數(shù)組中存在相同的值,通過(guò)id獲取的是相同的一個(gè)fragement片段,最后通過(guò)insertBefore操作DOM由于是相同的一個(gè)實(shí)例,故不會(huì)生效。
- child1
- child2
渲染的結(jié)果是child2在child1前面
使用track-by目的是自定義這個(gè)內(nèi)部的id,使得數(shù)組中具有相同的值的幾項(xiàng)都不會(huì)選擇到相同的實(shí)例,對(duì)于使用track-by='$index'還是其他唯一區(qū)分的id值有一定的區(qū)別,各有好處。
使用$index使得反轉(zhuǎn)的數(shù)據(jù)沒(méi)有移動(dòng)操作,而對(duì)于使用其他的id在順序不一樣的時(shí)候會(huì)有相應(yīng)的移動(dòng)操作。
對(duì)象的渲染
對(duì)象一般使用鍵作為內(nèi)部緩存對(duì)象的id,通過(guò)track-by也可以自定義這個(gè)id提高性能。
vm.model = { a: { id: 1, val: "model1"}, b: { id: 2, val: "model2"}, c: { id: 3, val: "model2"}, }
列表更新
vm.model = { d: { id: 1, val: "model1"}, e: { id: 2, val: "model2"}, f: { id: 3, val: "model2"} }
到此,相信大家對(duì)“vue使用源碼分析”有了更深的了解,不妨來(lái)實(shí)際操作一番吧!這里是創(chuàng)新互聯(lián)網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!