国产成人精品亚洲777人妖,欧美日韩精品一区视频,最新亚洲国产,国产乱码精品一区二区亚洲

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

詳解vue的hash跳轉(zhuǎn)原理

瀏覽:4日期:2022-10-02 17:06:39

在new vueRouter的時(shí)候我們可以傳入一個(gè)mode屬性,他可以接收三個(gè)值:hash/history/abstract

詳解vue的hash跳轉(zhuǎn)原理

hash和history的區(qū)別

history的路徑更美觀一點(diǎn) 比如http://yoursite.com/user/id,history是基于pushState()來完成 URL 跳轉(zhuǎn)而無須重新加載頁(yè)面。 但是強(qiáng)制刷新還是會(huì)有問題(服務(wù)端來解決這個(gè)問題),所以history模式需要后端人員配合使用。

hash的路徑會(huì)帶有#,比如http://yoursite.com#/user/id

HashHistory

class VueRouter{ constructor(options){ this.matcher = createMatcher(options.routes || []);//這里為了講解hash模式 所以就不進(jìn)行判斷用戶傳進(jìn)來的是哪種模式了 this.history = new HashHistory(this);//this vue-router的實(shí)例 }}

源碼這里創(chuàng)建了一個(gè)基類我們這里和源碼統(tǒng)一,這個(gè)基類封裝了三種模式公用的方法和屬性,那么我們?cè)谶@里創(chuàng)建一個(gè)HashHistory和基類History

import History from ’./base’// hash路由export default class HashHistory extends History{ constructor(router){ super(router); //繼承調(diào)用父類 等于call }}// 路由的基類export default class History { constructor(router){ this.router = router; }}

如果是hash路由,打開網(wǎng)站如果沒有hash默認(rèn)應(yīng)該添加#/

import History from ’./base’;function ensureSlash(){ if(window.location.hash){ return } window.location.hash = ’/’}export default class HashHistory extends History{ constructor(router){ super(router); ensureSlash(); // 確保有hash }}

再看一下初始化的邏輯(上面的router.init函數(shù))

init(app){ const history = this.history; // 初始化時(shí),應(yīng)該先拿到當(dāng)前路徑,進(jìn)行匹配邏輯 // 讓路由系統(tǒng)過度到某個(gè)路徑 const setupHashListener = ()=> { history.setupListener(); // 監(jiān)聽路徑變化 } history.transitionTo( // 父類提供方法負(fù)責(zé)跳轉(zhuǎn) history.getCurrentLocation(), // 子類獲取對(duì)應(yīng)的路徑 // 跳轉(zhuǎn)成功后注冊(cè)路徑監(jiān)聽,為視圖更新做準(zhǔn)備 setupHashListener )}

這里我們要分別實(shí)現(xiàn) transitionTo(基類方法)、 getCurrentLocation 、setupListener

getCurrentLocation實(shí)現(xiàn)

function getHash(){ return window.location.hash.slice(1);}export default class HashHistory extends History{ // ... getCurrentLocation(){ return getHash(); }}setupListener實(shí)現(xiàn)

export default class HashHistory extends History{ // ... setupListener(){ window.addEventListener(’hashchange’, ()=> { // 根據(jù)當(dāng)前hash值 過度到對(duì)應(yīng)路徑 this.transitionTo(getHash()); }) }}TransitionTo實(shí)現(xiàn)

export function createRoute(record, location) { // {path:’/’,matched:[record,record]} let res = []; if (record) { // 如果有記錄 while(record){ res.unshift(record); // 就將當(dāng)前記錄的父親放到前面 record = record.parent } } return { ...location, matched: res }}export default class History { constructor(router) { this.router = router; // 根據(jù)記錄和路徑返回對(duì)象,稍后會(huì)用于router-view的匹配 this.current = createRoute(null, { path: ’/’ }) } // 核心邏輯 transitionTo(location, onComplete) { // 去匹配路徑 let route = this.router.match(location); // 相同路徑不必過渡 if( location === route.path && route.matched.length === this.current.matched.length){ return } //更新路由并且下面會(huì)提到改變根實(shí)例上的_route屬性 this.updateRoute(route) onComplete && onComplete(); }}

export default class VueRouter{ // ... //做一個(gè)代理 match(location){ return this.matcher.match(location); }}macth方法

function match(location){ // 稍后根據(jù)路徑找到對(duì)應(yīng)的記錄 let record = pathMap[location] if (record) { // 根據(jù)記錄創(chuàng)建對(duì)應(yīng)的路由 //參數(shù):/about/a:{path:xx,component...},path:’/about/a’ return createRoute(record,{ path:location }) } // 找不到則返回空匹配 return createRoute(null, { path: location })}

我們不難發(fā)現(xiàn)路徑變化時(shí)都會(huì)更改current屬性,我們可以把current屬性變成響應(yīng)式的,每次current變化刷新視圖即可在install方法中

install(Vue) { Vue.mixin({ // 給所有組件的生命周期都增加beforeCreate方法 beforeCreate() { if (this.$options.router) { //調(diào)用Vue類中雙向數(shù)據(jù)綁定方法 Vue.util.defineReactive(this,’_route’,this._router.history.current); } } }); // $route和$router方法 這兩個(gè)方法僅僅是vue中最常見的代理 僅僅是為了更加方便 Object.defineProperty(Vue.prototype,’$route’,{ // 每個(gè)實(shí)例都可以獲取到$route屬性 get(){ return this._routerRoot._route;//上面剛進(jìn)行雙向數(shù)據(jù)綁定的 } }); Object.defineProperty(Vue.prototype,’$router’,{ // 每個(gè)實(shí)例都可以獲取router實(shí)例 get(){ return this._routerRoot._router; } }) }

切換路由每次初始化時(shí)都需要調(diào)用更新_route的方法,因?yàn)閕nstall的時(shí)候把_route進(jìn)行雙向數(shù)據(jù)綁定,剛進(jìn)來是沒有this._router.history.current的,通過發(fā)布訂閱方式來進(jìn)行訂閱和更新操作;在init方法中增加監(jiān)聽函數(shù)

history.listen((route) => { // 需要更新_route屬性,出入一個(gè)函數(shù) app._route = route});

export default class History { constructor(router) { // ... this.cb = null; } listen(cb){ this.cb = cb; // 注冊(cè)函數(shù) } updateRoute(route){ this.current =route; this.cb && this.cb(route); // 更新current后 更新_route屬性 }}

以上就是詳解vue的hash跳轉(zhuǎn)原理的詳細(xì)內(nèi)容,更多關(guān)于vue的hash跳轉(zhuǎn)原理的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Vue
相關(guān)文章:
主站蜘蛛池模板: 霍林郭勒市| 甘孜县| 舒兰市| 嵊州市| 宜君县| 平阳县| 舒兰市| 阿图什市| 冀州市| 定西市| 罗定市| 汝城县| 邵阳市| 盈江县| 福泉市| 札达县| 海林市| 垦利县| 新龙县| 光山县| 高台县| 桐城市| 行唐县| 宁化县| 漠河县| 肥东县| 灵石县| 大埔区| 隆昌县| 乾安县| 阿克苏市| 松江区| 社会| 诏安县| 武定县| 冀州市| 威宁| 扬中市| 巴彦淖尔市| 磐安县| 陆川县|